Главная страница
    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.046 c
14-1107207759
Logun
2005-02-01 00:42
2005.02.20
ВАС ИЗ ДАС .NET


14-1106902612
Antonn
2005-01-28 11:56
2005.02.20
Про мощные видяхи и хилые БП


14-1106953523
GanibalLector
2005-01-29 02:05
2005.02.20
Набережных С. - мастер!


1-1107344199
Bless
2005-02-02 14:36
2005.02.20
Overriding versus hiding


4-1104954636
Arnold
2005-01-05 22:50
2005.02.20
Как узнать список файлов в выбранной директории?





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
Английский Французский Немецкий Итальянский Португальский Русский Испанский