Форум программистов, компьютерный форум, киберфорум
Наши страницы

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
Schizorb
510 / 462 / 16
Регистрация: 07.04.2012
Сообщений: 869
Записей в блоге: 1
Завершенные тесты: 1
#1

Виртуальные функции - C++

08.12.2012, 11:37. Просмотров 524. Ответов 9
Метки нет (Все метки)

Вопрос. Как вызвать функцию наследника через указатель на базовый класс?

Так не компилируется
Код
3.cpp:19:19: error: invalid covariant return type for 'virtual Child Child::foo()'
3.cpp:7:18: error:   overriding 'virtual Base Base::foo()'
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
#include <iostream>
 
class Base
{
public:
  
    virtual Base foo()
    {
        std::cout << "BASE!!!\n";
        return *this;
    }
    
};
 
class Child : public Base
{
public:
 
    virtual Child foo()
    {
        std::cout << "CHILD!!!\n";
        return *this;
    }
    
};
 
 
int main()
{
    Child a, b;
    Base * test = &a;
    
    test -> foo();   
       
    return 0;
}
Если убираю virtual в базовом классе - компилируется, но вызывает функцию базового. Как решить проблему?
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
08.12.2012, 11:37
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Виртуальные функции (C++):

виртуальные и чисто виртуальные функции - C++
Чем они отличаются?? если можно, с примерами. И как из виртуальной функции сделать чисто виртуальную? #include &lt;iostream&gt; using...

виртуальные функции - C++
Добрый день. Начал изучать виртуальные функции и столкнулся с проблемой. Не могу перегрузить функцию. Объясните что не так и как нужно: ...

виртуальные функции - C++
ввести виртуальную функцию class A { void init(); } classB:A { void init(); }

Виртуальные функции. - C++
Приветствую всех. Дана такая программа (на самом деле она больше и сложнее, но структура и проблема те же): #include &lt;iostream&gt; ...

Виртуальные функции - C++
Какие происходят изменения, когда в классе объявляем одну или несколько функций виртуальными? Вот, например, если в SuperClass перед...

виртуальные функции - C++
сделать какой либо из методов класса виртуальным #include&lt;iostream.h&gt; #include&lt;stdio.h&gt; class chislo { int a; public: ...

9
Jupiter
Каратель
Эксперт С++
6564 / 3985 / 227
Регистрация: 26.03.2010
Сообщений: 9,273
Записей в блоге: 1
Завершенные тесты: 2
08.12.2012, 11:43 #2
возвращай ссылку или указатель
0
Schizorb
510 / 462 / 16
Регистрация: 07.04.2012
Сообщений: 869
Записей в блоге: 1
Завершенные тесты: 1
08.12.2012, 11:44  [ТС] #3
Цитата Сообщение от Jupiter Посмотреть сообщение
возвращай ссылку или указатель
Пробовал, так получается.

А что делать если объект создается внутри метода и его нужно вернуть?
0
Jupiter
Каратель
Эксперт С++
6564 / 3985 / 227
Регистрация: 26.03.2010
Сообщений: 9,273
Записей в блоге: 1
Завершенные тесты: 2
08.12.2012, 11:47 #4
создавай динамически
0
Schizorb
510 / 462 / 16
Регистрация: 07.04.2012
Сообщений: 869
Записей в блоге: 1
Завершенные тесты: 1
08.12.2012, 12:10  [ТС] #5
Ок. Теперь другая ситуация:

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
#include <iostream>
 
class Base
{
public:
  
    virtual void foo(const Base & x)
    {
        std::cout << "BASE!!!\n";
    }
    
};
 
class Child : public Base
{
public:
 
    virtual void foo(const Child & x)
    {
        std::cout << "CHILD!!!\n";
    }
    
};
Вызывается снова метод базового. Меняю на:

C++
1
virtual void foo(const Base & x)
Работает. Но мне внутри этой функции не нужна ссылка на базовый класс, я хочу работать с наследником. Как быть?
0
Jupiter
Каратель
Эксперт С++
6564 / 3985 / 227
Регистрация: 26.03.2010
Сообщений: 9,273
Записей в блоге: 1
Завершенные тесты: 2
08.12.2012, 12:21 #6
Цитата Сообщение от Schizorb Посмотреть сообщение
Вызывается снова метод базового.
потому что в классе Child создается новая виртуальная функция

Цитата Сообщение от Schizorb Посмотреть сообщение
C++
1
virtual void foo(const Base & x)
Работает. Но мне внутри этой функции не нужна ссылка на базовый класс, я хочу работать с наследником. Как быть?
вариант в лоб - кастуй к Child
0
Schizorb
510 / 462 / 16
Регистрация: 07.04.2012
Сообщений: 869
Записей в блоге: 1
Завершенные тесты: 1
08.12.2012, 17:57  [ТС] #7
Мда... заморочки еще те. Грубо говоря, мне всего навсего нужно создать операцию сложения в базовом и производном классах, чтобы она была виртуальной.

C++
1
virtual Base operator + (const Base & ) const;
Если возвращать указатель, то это какая-то ерунда. К тому же потом память очищать на стороне вызывающей функции. Как добиться простой записи, типа:

C++
1
2
Base a, b, c;
a = b + c;
Нарыл вот в одной из тем: Проблема с абстрактными классами и перегрузкой операторов

Не понял логику. Ну, создам я внешний оператор, ок. Как связать с виртуальными методами это дело?
0
Jupiter
Каратель
Эксперт С++
6564 / 3985 / 227
Регистрация: 26.03.2010
Сообщений: 9,273
Записей в блоге: 1
Завершенные тесты: 2
08.12.2012, 18:29 #8
Цитата Сообщение от Schizorb Посмотреть сообщение
Не понял логику. Ну, создам я внешний оператор, ок. Как связать с виртуальными методами это дело?
во внешнем операторе вызываем метод который выполняет сложение и метод этот виртуальный

Цитата Сообщение от Schizorb Посмотреть сообщение
Мда... заморочки еще те.
потому лучше выбирать что-то одно, либо операторы либо виртуальность методов, код должен быть простым и понятным
0
ValeryS
Модератор
6744 / 5153 / 492
Регистрация: 14.02.2011
Сообщений: 17,320
08.12.2012, 18:30 #9
Цитата Сообщение от Schizorb Посмотреть сообщение
return *this;
ты это что хочешь вернуть
Цитата Сообщение от Schizorb Посмотреть сообщение
virtual Child foo()
* * {
* * * * std::cout << "CHILD!!!\n";
* * * * return *this;
* * }
Цитата Сообщение от Schizorb Посмотреть сообщение
virtual Base foo()
* * {
* * * * std::cout << "BASE!!!\n";
* * * * return *this;
* * }
C++
1
2
3
4
5
6
7
8
9
10
11
virtual Base* foo()
 {
 std::cout << "BASE!!!\n";
 return this;
 }
 
virtual Base* foo()
 {
 std::cout << "CHILD!!!\n";
 return  this;
}
0
OhMyGodSoLong
~ Эврика! ~
1244 / 993 / 42
Регистрация: 24.07.2012
Сообщений: 2,002
08.12.2012, 20:13 #10
Цитата Сообщение от Schizorb Посмотреть сообщение
Мда... заморочки еще те. Грубо говоря, мне всего навсего нужно создать операцию сложения в базовом и производном классах, чтобы она была виртуальной.
Это называется множественная диспетчеризация. Потому что вы хотите вызывать метод, зависящий от runtime-типов двух аргументов. Ведь когда есть такое:
C++
1
2
3
class Number { /* ... */ };
class Integer : public Number { /* ... */ };
class Ratinal : public Number { /* ... */ };
и пишется такое:
C++
1
2
Number *a = new Rational(2, 3), *b = new Integer(5);
Number *c = *a + *b; // *c == Rational(17, 3)
То operator+ зависит как от типа a, так и от типа b.

Для Си++, в общем, есть три метода реализации такого поведения:
  • много dynamic_cast;
  • сделать свою двухмерную таблицу виртуальных функций с го и гейшами;
  • double dispatch (aka паттерн visitor).
Примеры в той же вики можете глянуть. (Пыщь, пыщь, пыщь.) Да, это сложно, муторно, тормозит и требует вагон boilerplate-кода, но за удобства надо платить.

Цитата Сообщение от Schizorb Посмотреть сообщение
Если возвращать указатель, то это какая-то ерунда. К тому же потом память очищать на стороне вызывающей функции.
А придётся. Вот в частности хотелки такого рода и дали жизнь всяким смарт-поинтерам.
0
08.12.2012, 20:13
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
08.12.2012, 20:13
Привет! Вот еще темы с ответами:

Виртуальные функции С++ - C++
Задача: В классе хранится целое, и определяется виртуальная функция shownum(). Создать 2 производных класса, наследующие класс num. В...

Виртуальные функции - C++
Здрасте! вот задачка, какбэ сделал, но нужны виртуальные функции. даже не знаю как их сделать. #include &lt;iostream.h&gt; #include...

Виртуальные функции - C++
Доброго времени суток. Есть задание. Создать абстрактный класс Function (функция) с виртуальными методами вычисления значения функции...

виртуальные функции - C++
C++ подскажите пожалуста на словах код писать ненадо. как преопределить функцию virtual Print(). Создать абстрактный базовый класс...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
10
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2018, vBulletin Solutions, Inc.