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

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

Восстановить пароль Регистрация
 
moskitos80
 Аватар для moskitos80
39 / 39 / 0
Регистрация: 04.10.2011
Сообщений: 128
07.08.2012, 12:16     Не вызывается нужный конструктор базового класса #1
Здравствуйте. Я пока в процессе изучения С++. Имеется следующая структура классов:

Базовый: 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
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
John Prick
754 / 687 / 123
Регистрация: 27.07.2012
Сообщений: 1,974
Завершенные тесты: 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), поэтому компилятор потребовал от тебя конструктор по умолчанию.
DiffEreD
 Аватар для DiffEreD
1420 / 757 / 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;
    }
};
John Prick
754 / 687 / 123
Регистрация: 27.07.2012
Сообщений: 1,974
Завершенные тесты: 3
07.08.2012, 12:46     Не вызывается нужный конструктор базового класса #4
Если что: С.Дьхерст "Скользкие места С++", Совет 53. Инициализация виртуальных базовых классов (стр.124).
moskitos80
 Аватар для moskitos80
39 / 39 / 0
Регистрация: 04.10.2011
Сообщений: 128
07.08.2012, 12:54  [ТС]     Не вызывается нужный конструктор базового класса #5
Цитата Сообщение от John Prick Посмотреть сообщение
Это связано с тем, что виртуальные наследники от Person, классы Manager и Student не инициализируют поля Person. Это должен делать невиртуальный наследник - Employee.
Отлично! Чётко и по существу - теперь всё ясно. Я предполагал что здесь дело в виртуальных наследниках. Отдельно спасибо за С.Дьхерста
yuron_477 - я бы уже и сам догодался, но всё же спасибо и Вам.

Ещё маленький вопрос: в конструкторах у виртуальных наследников Manager и Student вообще не имеет смысла вызывать конструктор базового класса Person ?
John Prick
754 / 687 / 123
Регистрация: 27.07.2012
Сообщений: 1,974
Завершенные тесты: 3
07.08.2012, 13:02     Не вызывается нужный конструктор базового класса #6
Цитата Сообщение от moskitos80 Посмотреть сообщение
спасибо за С.Дьхерста
Я опечатался, и вы в след за мной. Стивен Дьюхерст (Stephen Dewhurst), если будете искать.
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
07.08.2012, 13:13     Не вызывается нужный конструктор базового класса
Еще ссылки по теме:

Конструктор вызывается 2 раза C++
Конструктор базового класса C++
Наследование: почему вызывается метод базового класса, а не производного? C++

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

Или воспользуйтесь поиском по форуму:
moskitos80
 Аватар для moskitos80
39 / 39 / 0
Регистрация: 04.10.2011
Сообщений: 128
07.08.2012, 13:13  [ТС]     Не вызывается нужный конструктор базового класса #7
Цитата Сообщение от John Prick Посмотреть сообщение
Я опечатался, и вы в след за мной. Стивен Дьюхерст (Stephen Dewhurst), если будете искать.
Да-да поисковик меня уже поравил.

А по-поводу того что:
Ещё маленький вопрос: в конструкторах у виртуальных наследников Manager и Student вообще не имеет смысла вызывать конструктор базового класса Person ?
Сам отвечу: по-ходу дела можно и не вызывать, но тогда опять же для них будет вызываться конструктор класса Person без параметров, а так как у базового класса Person уже явно определён конструктор с параметром, то конструктор без параметра тоже потребуется определить явно. Но тем не менее будет работать. Ох уж эти конструкторы
Yandex
Объявления
07.08.2012, 13:13     Не вызывается нужный конструктор базового класса
Ответ Создать тему
Опции темы

Текущее время: 19:27. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2016, vBulletin Solutions, Inc.
Рейтинг@Mail.ru