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

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

Войти
Регистрация
Восстановить пароль
 
moskitos80
61 / 61 / 10
Регистрация: 04.10.2011
Сообщений: 217
Завершенные тесты: 1
#1

Не вызывается нужный конструктор базового класса - C++

07.08.2012, 12:16. Просмотров 494. Ответов 6
Метки нет (Все метки)

Здравствуйте. Я пока в процессе изучения С++. Имеется следующая структура классов:

Базовый: Person,
расширяющие его: Student и Manager,
и наследующий их обоих класс Employee

- древо в виде ромба Вот собственно код:

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
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
#include <iostream>
#include <iomanip>
#include <string.h>
 
using namespace std;
 
///////////////////////////////////////////////////////////
class Person {
 
      protected :
          string name;
 
      public :
          Person() // Этого конструктора в начале не было.
          {}
          Person(string nm) :
                name (nm)
          {}
          virtual ~Person()
          {
            cout << "Person destruct" << endl;
          }
 
          virtual void getInfo() = 0;
};
class Student : virtual public Person {
 
      protected :
          int avg;
          int course;
 
      public :
          Student(string nm, int cr = 1, int g = 0) :
                Person(nm),
                avg (g),
                course (cr)
          {}
          ~Student()
          {
              cout << "Student destruct" << endl;
          }
};
class Manager : virtual public Person {
 
      protected :
          float salary;
          string post;
 
      public :
          Manager(string nm, float sal = 0.0f, string pst = "junior") :
                Person(nm),
                post (pst),
                salary (sal)
          {}
          ~Manager()
          {
              cout << "Manager destruct" << endl;
          }
};
class Employee : public Manager, public Student {
 
      public :          
          Employee(string nm) :
                         Manager(nm),
                         Student(nm)
          {}
          ~Employee()
          {
              cout << "Employee destruct" << endl;
          }
          void getInfo()
          {
              cout << "Manager: "      << endl
                   << "name  " << name << endl
                   << "post "  << post << endl;
              if (course > 0) {
                  cout << "It is also a student of " << course << " course " << endl;
              }
              cout << setfill('.') << setw(80) << "" << endl;
          }
};
///////////////////////////////////////////////////////////
 
int main() {
    string data;
 
    cout << "Enter name "; cin >> data; // Ввод имени
    cout << endl;
 
    Person* ptr = new Employee(data);
    ptr->getInfo();
 
    delete ptr;
    return 0;
}
Для того чтобы обратиться к полю name из Employee::getInfo() (строка 75) мне пришлось сделать следующее:

class Manager : virtual public Person ...
class Student : virtual public Person ...

что бы убрать неоднозначность, но после этого в строке 66 (конструктор Employee) компилятор стал требовать явно указать конструктор без аргументов для базового класса Person:

error: no matching function for call to 'Person::Person()'|

когда же я его добавил (строка 15) все стало ок, но теперь при работе программы:


Enter name (вводим: Vasya)

Manager:
name --- ???
post junior
It is also a student of 1 course
...............................................................................

Employee destruct
Student destruct
Manager destruct
Person destruct

Process returned 0 (0x0) execution time : 9.386 s
Press any key to continue.


Нету введённого нами имени! - я понимаю, что для базового класса Person вызывается конструктор без аргументов, который потребовал компилятор, но не понимаю почему он его потребовал, ведь у меня нигде нет ситуации, способствующей вызову Person() и почему не вызывается Person(string nm) - который мне нужен для инициализации поля name
0
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
07.08.2012, 12:16
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Не вызывается нужный конструктор базового класса (C++):

Не вызывается конструктор базового класса - C++
Доброго времени суток. У меня проблема:в производном классе не вызывается конструктор базового класса. Думаю описывать класс не стоит,...

При создании класса конструктор вызывается 2 раза, затем вызывается деструктор о_О - C++
Вот такой кодclass A { public: A(){} virtual ~A(){} }; class C { public:

Наследование: почему вызывается метод базового класса, а не производного? - C++
Всем привет, такой вопрос, почему вызывается в данном коде метод базового класса, а не производного (без указания ключевого слова virtual...

Конструктор базового класса - C++
Привет, друзья, есть, к примеру, такие классы: class Base { public: int x; Base() { } };

Конструктор базового класса - C++
Такая проблема: имеется конструктор базового класса ErrorDataErrorData(ErrorCod cod, bool isFatal);, где ErrorCod вот такая...

Будет ли утрачена память, когда конструктор копирования вызывается для уже существующего объекта класса? - C++
class A { char * v; A(); ~A(); A(const A &amp;obj); } ///////////////////// A::A() {

Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
John Prick
801 / 734 / 145
Регистрация: 27.07.2012
Сообщений: 2,107
Завершенные тесты: 3
07.08.2012, 12:39 #2
Не уверен, но по-моему, конструктор Employee должен выглядеть так:
C++
1
2
3
4
Employee(string nm) :
                         Person(nm),
                         Manager(nm),
                         Student(nm)
Это связано с тем, что виртуальные наследники от Person, классы Manager и Student не инициализируют поля Person. Это должен делать невиртуальный наследник - Employee. В списке инициализации ты не указал Person(nm), поэтому компилятор потребовал от тебя конструктор по умолчанию.
0
DiffEreD
1430 / 767 / 95
Регистрация: 21.06.2011
Сообщений: 1,740
Записей в блоге: 2
07.08.2012, 12:40 #3
Попробуйте
так
C++ (Qt)
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
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
class Person {
 
protected :
    string name;
 
public :
    //Person() // Этого конструктора в начале не было.
    //{}
    Person(string nm) :
        name (nm)
    {}
    virtual ~Person()
    {
        cout << "Person destruct" << endl;
    }
 
    virtual void getInfo() = 0;
};
class Student : virtual public Person {
 
protected :
    int avg;
    int course;
 
public :
    Student(string nm, int cr = 1, int g = 0) :
        Person(nm),
        avg (g),
        course (cr)
    {}
    ~Student()
    {
        cout << "Student destruct" << endl;
    }
};
class Manager : virtual public Person {
 
protected :
    float salary;
    string post;
 
public :
    Manager(string nm, float sal = 0.0f, string pst = "junior") :
        Person(nm),
        post (pst),
        salary (sal)
    {}
    ~Manager()
    {
        cout << "Manager destruct" << endl;
    }
};
class Employee : public Manager, public Student {
 
public :          
    Employee(string nm) :
        Manager(nm),
        Student(nm),
        Person(nm)
    {}
    ~Employee()
    {
        cout << "Employee destruct" << endl;
    }
    void getInfo()
    {
        cout << "Manager: "      << endl
            << "name  " << name << endl
            << "post "  << post << endl;
        if (course > 0) {
            cout << "It is also a student of " << course << " course " << endl;
        }
        cout << setfill('.') << setw(80) << "" << endl;
    }
};
1
John Prick
801 / 734 / 145
Регистрация: 27.07.2012
Сообщений: 2,107
Завершенные тесты: 3
07.08.2012, 12:46 #4
Если что: С.Дьхерст "Скользкие места С++", Совет 53. Инициализация виртуальных базовых классов (стр.124).
1
moskitos80
61 / 61 / 10
Регистрация: 04.10.2011
Сообщений: 217
Завершенные тесты: 1
07.08.2012, 12:54  [ТС] #5
Цитата Сообщение от John Prick Посмотреть сообщение
Это связано с тем, что виртуальные наследники от Person, классы Manager и Student не инициализируют поля Person. Это должен делать невиртуальный наследник - Employee.
Отлично! Чётко и по существу - теперь всё ясно. Я предполагал что здесь дело в виртуальных наследниках. Отдельно спасибо за С.Дьхерста
yuron_477 - я бы уже и сам догодался, но всё же спасибо и Вам.

Ещё маленький вопрос: в конструкторах у виртуальных наследников Manager и Student вообще не имеет смысла вызывать конструктор базового класса Person ?
0
John Prick
801 / 734 / 145
Регистрация: 27.07.2012
Сообщений: 2,107
Завершенные тесты: 3
07.08.2012, 13:02 #6
Цитата Сообщение от moskitos80 Посмотреть сообщение
спасибо за С.Дьхерста
Я опечатался, и вы в след за мной. Стивен Дьюхерст (Stephen Dewhurst), если будете искать.
0
moskitos80
61 / 61 / 10
Регистрация: 04.10.2011
Сообщений: 217
Завершенные тесты: 1
07.08.2012, 13:13  [ТС] #7
Цитата Сообщение от John Prick Посмотреть сообщение
Я опечатался, и вы в след за мной. Стивен Дьюхерст (Stephen Dewhurst), если будете искать.
Да-да поисковик меня уже поравил.

А по-поводу того что:
Ещё маленький вопрос: в конструкторах у виртуальных наследников Manager и Student вообще не имеет смысла вызывать конструктор базового класса Person ?
Сам отвечу: по-ходу дела можно и не вызывать, но тогда опять же для них будет вызываться конструктор класса Person без параметров, а так как у базового класса Person уже явно определён конструктор с параметром, то конструктор без параметра тоже потребуется определить явно. Но тем не менее будет работать. Ох уж эти конструкторы
0
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
07.08.2012, 13:13
Привет! Вот еще темы с ответами:

this(Всегда ли вызывается конструктор при не явной передачи объекта в конструктор) - C++
class Test { int i; public: void test(int i) { this -&gt; i = i; } }; Конструктор Test::Test всегда вызывается...

Конструктор класса не видит конструктор по умолчанию другого класса - C++
Ошибка, естественно, в Classes.cpp, в строке 20. Ругается, что у класса TailNode нет конструктора по умолчанию, хотя он там, конечно, есть....

Не вызывается нужный элемент массива по значению управляющей переменной в цикле for - C++
Вот часть кода : for(int i=0; i &lt; 5; i++){ cout &lt;&lt; &quot;Этот объект &quot; &lt;&lt; shapes-&gt;getName() &lt;&lt; &quot;\n&quot;; cout &lt;&lt; &quot;Площадь равна &quot; &lt;&lt; ...

Не вызывается конструктор - C++
Есть конструктор, который я вызываю если, число, которое я передаю больше 20. Конструктор должен к нулю приравнять значение, но почему то...


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

Или воспользуйтесь поиском по форуму:
Yandex
Объявления
07.08.2012, 13:13
Ответ Создать тему
Опции темы

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