Главная страница
    Top.Mail.Ru    Яндекс.Метрика
Текущий архив: 2005.02.20;
Скачать: CL | DM;

Вниз

Overriding versus hiding   Найти похожие ветки 

 
Bless ©   (2005-02-02 14:36) [0]

Процитирую целиком текст справки из сабжевого раздела.


If a method declaration specifies the same method identifier and parameter signature as an inherited method, but doesn"t include override, the new declaration merely hides the inherited one without overriding it. Both methods exist in the descendant class, where the method name is statically bound. For example,


type
 T1 = class(TObject)
   procedure Act; virtual;
 end;
 T2 = class(T1)
   procedure Act;  // Act is redeclared, but not overridden
 end;
var
 SomeObject: T1;
begin
 SomeObject := T2.Create;
 SomeObject.Act;  // calls T1.Act
end;


Объясните пожалуйста, почему строчка SomeObject.Act приводит к вызову метода Act из класса T1?


 
REA   (2005-02-02 14:42) [1]

Компилятор знает, что SomeObject: T1 и вызывает метод T1.Act. Про метод T2.Act ему не известно в контексте этого объекта.


 
icWasya ©   (2005-02-02 15:24) [2]

если написать так
type
T1 = class(TObject)
  procedure Act; virtual;
end;
T2 = class(T1)
  procedure Act;overridde; end;
var
SomeObject: T1;
begin
SomeObject := T2.Create;
SomeObject.Act;  // calls T2.Act
end;

то будет щастье


 
Bless ©   (2005-02-02 16:09) [3]

REA[1]>
Компилятор знает, что SomeObject: T1 и вызывает метод T1.Act. Про метод T2.Act ему не известно в контексте этого объекта.

Как это ничего не известно? SomeObject.ClassType=T2.

icWasya ©  (02.02.05 15:24) [2]
то будет щастье


Это я и сам знаю. Только я не об этом спрашивал.


 
REA   (2005-02-02 16:52) [4]

SomeObject.ClassType это на момент выполнения. А на момент компиляции неизвестно, что попадет в эту переменную, поэтому компилятор не делает предположений относительно содержимого переменной, а вызывает метод по известному адресу. К тому же ему это ни к чему - метод не перекрыт и имеет право на жизнь.
Компилятор вызывает метод не пользуясь таблицей виртуальных методов - т.е. по фиксированному смещению, а не смещению прописанному в таблице виртуальных методов.


 
Владислав ©   (2005-02-02 17:03) [5]

Судя по всему, Вам нужно познакомиться с разницей в вызовах виртуальных и невиртуальных методов.


 
Bless ©   (2005-02-02 17:48) [6]

Владислав ©  (02.02.05 17:03) [5]

Кстати, "Вы" - это черезчур :)


 
Bless ©   (2005-02-02 18:09) [7]

Вот блин, потерялся один пост.
Повторяю:

>Владислав ©  (02.02.05 17:03) [5]
Судя по всему, Вам нужно познакомиться с разницей в вызовах виртуальных и невиртуальных методов.

Это вряд ли (хотя чем черт не шутит). Ведь если поменять в посте [0]

T2 = class(T1)
 procedure Act;  // Act is redeclared, but not overridden
end;


на


T2 = class(T1)
 procedure Act; virtual; // Act is redeclared, but not overridden
end;


то результат останется тем же.


REA  (02.02.05 16:52) [4]
А на момент компиляции неизвестно, что попадет в эту переменную, поэтому компилятор не делает предположений относительно содержимого переменной, а вызывает метод по известному адресу.


Ну пускай даже компилятору неизвестно, что попадет в эту переменную. Но как он может подставить вызов метода по известному адресу, если этот метод - виртуальный? Что-то я сильно сомневаюсь, что компилятор вдруг решил обратиться к виртуальному методу не по таблице виртуальных методов а по фиксированному смещению. Это есть в справке?


 
Владислав ©   (2005-02-02 18:37) [8]

Ну пускай даже компилятору неизвестно, что попадет в эту переменную. Но как он может подставить вызов метода по известному адресу, если этот метод - виртуальный? Что-то я сильно сомневаюсь, что компилятор вдруг решил обратиться к виртуальному методу не по таблице виртуальных методов а по фиксированному смещению. Это есть в справке?

В точку. Просто в [4] некорректное утверждение.
Вот у тебя и вызывается виртуальный метод T1. Потому что в таблице именно он и есть. Если бы написал override, то там был бы метод T2.

Еще раз прочти [5].


 
Bless ©   (2005-02-03 09:10) [9]

>Еще раз прочти [5]

Зачем? Причем тут статические методы? Я же написал (и проверил, кстати), что если метод T2.act сделать виртуальным, то ничего не изменится.

>Вот у тебя и вызывается виртуальный метод T1. Потому что в
>таблице именно он и есть. Если бы написал override, то там был бы
>метод T2.

Так он и так там есть! Не в виртуальной таблице правда, поскольку он статический, но все-равно есть. И его тоже можно вызвать. Например, вот так: Т2(SomeObject).Act. Но почему сразу вызывается не он? Как компилятор рассуждает?

Для простоты переделаю исходный пример:

type
T1 = class(TObject)
 procedure Act; virtual;
end;
T2 = class(T1)
 procedure Act; VIRTUAL; //!!!!!!!!!!!!!
end;
var
SomeObject: T1;
begin
SomeObject := T2.Create;
SomeObject.Act;  // calls T1.Act
end;


Теперь в таблице виртуальных методов класса T2 присутствуют два метода Act (тоже проверил): один унаследованный и один свой. Почему в примере вызвается унаследованный?


 
Владислав ©   (2005-02-03 09:14) [10]

Знаешь, я больше повторяться не буду. Вот если после прочтения соответствующей документации у тебя появится желание продолжить тему, с удовольствием поучаствую.


 
REA   (2005-02-03 09:59) [11]

Это разные методы. Ты не заменил метод (override) в таблице, а добавил второй с тем же именем. По-хорошему надо бы блокировать такие попытки, но компилятор видимо надеется, что программист знает, что делает.



Страницы: 1 вся ветка

Текущий архив: 2005.02.20;
Скачать: CL | DM;

Наверх




Память: 0.48 MB
Время: 0.041 c
8-1093626380
ИМХО
2004-08-27 21:06
2005.02.20
DXF


14-1107348880
Vlad Oshin
2005-02-02 15:54
2005.02.20
у кого нибудь компьютер W95 входит в домен W2000?


1-1107331582
MakedoneZ
2005-02-02 11:06
2005.02.20
Почему неоткрывается frmGraph?


3-1106213481
SERGGG
2005-01-20 12:31
2005.02.20
SQL BUILDER


14-1106953426
Narik
2005-01-29 02:03
2005.02.20
Скачайте Demo Delphi 2005





Afrikaans Albanian Arabic Armenian Azerbaijani Basque Belarusian Bulgarian Catalan Chinese (Simplified) Chinese (Traditional) Croatian Czech Danish Dutch English Estonian Filipino Finnish French
Galician Georgian German Greek Haitian Creole Hebrew Hindi Hungarian Icelandic Indonesian Irish Italian Japanese Korean Latvian Lithuanian Macedonian Malay Maltese Norwegian
Persian Polish Portuguese Romanian Russian Serbian Slovak Slovenian Spanish Swahili Swedish Thai Turkish Ukrainian Urdu Vietnamese Welsh Yiddish Bengali Bosnian
Cebuano Esperanto Gujarati Hausa Hmong Igbo Javanese Kannada Khmer Lao Latin Maori Marathi Mongolian Nepali Punjabi Somali Tamil Telugu Yoruba
Zulu
Английский Французский Немецкий Итальянский Португальский Русский Испанский