November 2019

S M T W T F S
      12
34 5 678 9
10111213141516
17181920212223
24252627282930

Style Credit

Expand Cut Tags

No cut tags
Thursday, March 31st, 2005 04:49 pm


Берём простейший код:
class B {
  public:
    void method(char c);
    void method(int i);
};

class M {
 int i;
};

class D: public B {
  public:
    void method(M);
};

int main() {
  D d;
  M m;

  d.method('c'); // не работает
  d.method(0); // тоже не работает
  d.method(m); // работает
}

И обнаруживаем, что для объекта производного класса методы базового нифига не вызываются (error: no matching function for call to `D::method(char)'), потому что они перекрыты одноимённым методом в производном.
И где, спрашивается, это ваше хвалёное наследование?
Да, если из D убрать одноимённый метод, то всё работает.
Кто знает, чем обусловлена такая странность?
Thursday, March 31st, 2005 02:19 pm (UTC)
Я чую, из курилки в вашу комнату дым от неслабой травы заносит.
То, что ты нарисовал, называется hiding, оно же - сокрытие, и вступает в силу, если есть коллизия имён у потомков и предков. Для наследования необходимо, чтобы а) у фнукций были идентичные сигнатуры и наличие где-нибудь (где тебе больше нравится, но я обычно пишу в начале строки, где объявляю функцию) ключевого слова virtual, или б) отсутствие символов с таким же именем в классе наследнике.
Если же ты хочешь из класса предка вытащить символы и сделать их доступными в классе потомке, то надо действительно использовать ключевое слово using.
Теперь про то, чем обусловленно: представь, что все символы из класса предка экспортируются в класс наследник. Ты не можешь отказаться от этого поведения. Если ты хочешь ввести в потомке символ, разрешаемый по более общему типу, то не можешь этого сделать. Ты не сможешь отловить на этапе компиляции целую кучу ошибок. В существующем порядке ты можешь сам написать, какие символы импортировать.
Thursday, March 31st, 2005 02:43 pm (UTC)
1) Коллизии не наблюдаю. Mangled имена методов абсолютно разные получаются.

а) virtual тут вовсе ни при чём, я про полиморфизм ничего не гворил, по ссылке ничего не вызывал, я хочу просто для объекта вызывать метод из его класса, если он есть, а если нет - то унаследованный от базового.

2) не понимаю, чем принципиально отличается перегрузка невиртуальных методов в пределах одного класса и в базовом и порождённом. Пойдем на кухню, ты мне на пальцах покажешь


Thursday, March 31st, 2005 03:24 pm (UTC)
Overloading вообще для безопасности не работает across different scopes.
Т.е. вот такой пример есть в страуструпе:

void f(int);
void g()
{
void f(double);
f(1) ; // call f(double)
}

И в принципе идея здравая нужно сказать.
Friday, April 1st, 2005 08:32 pm (UTC)
> Пойдем на кухню, ты мне на пальцах покажешь
11 апреля, ок?