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

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

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 14, средняя оценка - 4.86
KeyGen
384 / 291 / 6
Регистрация: 07.08.2011
Сообщений: 790
Записей в блоге: 1
#1

Производный класс. Перегрузка << (cout). - C++

18.01.2012, 20:35. Просмотров 1930. Ответов 26
Метки нет (Все метки)

При инициализации массива указателей базового класса при вызове cout производный класс не отображается, отображается базовый. Без указателей все в порядке. Подскажите пожалуйста почему.
Вот пример кода:

bor.h
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
#include <iostream>
 
class Port
{
private:
    char * brand;   // название
    char * style;   // тип
    int bottles;    // бутылки
protected:
    const char * Brand() const {return brand;}
    const int Bottles() const {return bottles;} 
public:
    Port();
    Port(const char * br, const char * st, int b);
    Port(const Port & p);
    virtual~Port();
    Port & operator = (const Port & p);
    Port & operator += (int b);
    Port & operator -= (int b);
    int BottleCont()const {return bottles;}
    virtual void Show() const;
    friend std::ostream & operator<<(std::ostream & os, const Port & p);
};
 
class VintagePort : public Port
{
private:
    char * nickname;    // старый или новый
    int year;           // год
public:
    VintagePort();
    VintagePort(const char *br, int b, const char *nn, int y);
    VintagePort(const VintagePort & vp);
    ~VintagePort();
    VintagePort & operator= (const VintagePort & vp);
    void Show() const;
    friend std::ostream & operator<<(std::ostream & os, const VintagePort & vp);
};
bor.cpp
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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
#include "bor.h"
 
// методы Port
Port::Port()
{
    char *br = "none";
    brand = new char[strlen(br)+1];
    strcpy(brand,br);
    style = new char[strlen(br)+1];
    strcpy(style,br);
    bottles = 0;    
}
Port::Port(const char * br, const char * st, int b)
{
    brand = new char[strlen(br)+1];
    strcpy(brand,br);
    style = new char[strlen(st)+1];
    strcpy(style,st);
    bottles = b;    
}
Port::Port(const Port & p)
{
    brand = new char[strlen(p.brand)+1];
    strcpy(brand,p.brand);
    style = new char[strlen(p.style)+1];
    strcpy(style,p.style);
    bottles = p.bottles;
}
Port & Port::operator = (const Port & p)
{
    if(this == &p)
        return *this;
    delete [] brand;
    delete [] style;
    brand = new char[strlen(p.brand)+1];
    strcpy(brand,p.brand);
    style = new char[strlen(p.style)+1];
    strcpy(style,p.style);
    bottles = p.bottles;
 
    return *this;
}
Port & Port::operator += (int b)
{
    bottles+=b;
    return *this;
}
Port & Port::operator -= (int b)
{
    bottles-=b;
    return *this;
}
void Port::Show() const
{
    std::cout << "Brand: " << brand << std::endl;
    std::cout << "Kind: " << style << std::endl;
    std::cout << "Bottles: " << bottles;
}
std::ostream & operator<<(std::ostream & os, const Port & p)
{
    os << p.brand << ", " << p.style << ", " << p.bottles;
 
    return os;
}
 
Port::~Port() 
{
    delete [] brand; 
    delete [] style; 
}
 
// методы VintagePort
VintagePort::VintagePort():Port()
{
    nickname = new char [5];
    strcpy(nickname,"none");
    year = 0;
}
 
VintagePort::VintagePort(const char *br, int b, const char *nn, int y):Port(br,"none",b)
{
    nickname = new char[strlen(nn)+1];
    strcpy(nickname,nn);
    year = y;
}
VintagePort::VintagePort(const VintagePort & vp) : Port(vp)
{
    nickname = new char[strlen(vp.nickname)+1];
    strcpy(nickname,vp.nickname);
    year = vp.year;
}
VintagePort & VintagePort::operator= (const VintagePort & vp)
{
    if(this == &vp)
        return *this;
    Port::operator=(vp);
    delete [] nickname;
    nickname = new char[strlen(vp.nickname)+1];
    strcpy(nickname,vp.nickname);
    year = vp.year;
 
}
void VintagePort::Show() const
{
    std::cout << "Название: " << Brand() << std::endl;
    std::cout << "Kоличество бутылок: " << Bottles() << std::endl;
    std::cout << "Старое или новое: " << nickname << std::endl;
    std::cout << "Год: " << year;
}
std::ostream & operator<<(std::ostream & os, const VintagePort & vp)
{
    os << (const Port &)vp;
    os << ", " << vp.nickname << ", " << vp.year;
    return os;
}
 
VintagePort::~VintagePort() 
{
    delete [] nickname;
}
main.cpp
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
#include <iostream>
#include "bor.h"
 
const int SIZE = 2;
 
int main()
{
    setlocale (LC_ALL, "rus");
    using std::cout;
    using std::endl;
    using std::cin;
 
            char ch[40];
            char ch2[40];
            int reiting;
            int reiting2;
            int i;
 
    Port *mass [SIZE];
 
    for(i=0; i<SIZE; i++)
    {
        char letter;
        if(i!=0)
            cout << endl;
        cout << "Введите что пишем:\nPort - A, VintagePort - B, Выход - С: ";
        cin >> letter;
        if((letter<65||letter>67)&&(letter<97||letter>99))
        {i--; cin.getline(ch,40); continue;}
        cin.get();
        if(letter == 'C' || letter == 'c')
            break;
 
        switch(letter)
        {
        case 'A':
        case 'a':
            cout << "\nPort:\n";
            cout << "Введите название: ";
            cin.getline(ch,40);
            cout << "Введите тип: ";
            cin.getline(ch2,40);
            cout << "Введите количества бутылок: ";
            cin>>reiting;
 
            mass[i] = new Port(ch,ch2,reiting);
            break;
        case 'B':
        case 'b':
            cout << "\nVintagePort:\n";
            cout << "Введите название: ";
            cin.getline(ch2,40);
            cout << "Введите количество бутылок: ";
            cin >> reiting2;
            cin.get();
            cout << "Введите старый или новый: ";
            cin.getline(ch,40);
            cout << "Введите год: ";
            cin>>reiting;
 
            mass[i] = new VintagePort(ch2,reiting2,ch,reiting);
            break;
        }
    }
    
    if(i!=0)
    {
    cout << endl << endl;
    for(int i=0; i<SIZE; i++){
        mass[i]->Show();
    cout << endl << endl;
    }
 
    //cout << *(mass[0]) << endl;   // отображает Port в любом случае
 
    for(int j=0; j<SIZE; j++)
    {
        delete mass[j];
    }
    }
 
    cout << "By :)\n\n";
 
    system("pause");
    return 0;
}
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
18.01.2012, 20:35
Здравствуйте! Я подобрал для вас темы с ответами на вопрос Производный класс. Перегрузка << (cout). (C++):

Разработать производный от абстрактного класса Figure класс и класс, производный от производного - C++
Разработать базовый абстрактный класс Figure с пустым виртуальным методом print для вывода в дальнейшем полной информации об объекте: ...

Есть производный класс и базовый; при чём производный использует только ЧАСТЬ ресурсов базового, правильно ли это? - C++
Ну то есть базовый класс A, с полями, допустим, a, b, c, d. И я проектирую производный класс B и я железно знаю, что он будет использовать...

Создать класс Point и производный класс ColoredPoint (цветная точка) - C++
Создайте класс Point (точка на плоскости), характеризующийся двумя координатами. Создайте производный класс ColoredPoint (цветная точка), в...

Разработать класс Tableware (посуда) и производный класс Dish (тарелка). Описать атрибуты - C++
Разработать класс Tableware (посуда) и производный класс Dish (тарелка). Описать атрибуты.

Разработать класс Man (человек) и производный класс Student (студент). Описать атрибуты. - C++
Разработать класс Man (человек) и производный класс Student (студент). Описать атрибуты.

Создать базовый класс - Array и производный класс - Money для работы денежной суммы - C++
ПОМОГИТЕ, ПОЖАЛУЙСТА, С ЗАДАЧЕЙ Создать базовый класс - Array и производный класс - Money для работы денежной суммы

26
DU
1483 / 1129 / 45
Регистрация: 05.12.2011
Сообщений: 2,279
18.01.2012, 23:11 #16
И ходя выше задекларировано, что mass[i] указывает на Port, но происходит переопределение типа и происходит корректно, надо сказать, ибо Port- родителський класс. Так что всё просто; с этого момента mass [i] (mass[1]) указывает на винтажный порт!
да не так все. вы похоже сами проблему не понимаете.
вот упрощенный пример. соберите и посмотрите как все работает
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
#include <iostream>
 
namespace Bad
{
  class Base
  {
  };
 
  std::ostream& operator << (std::ostream& out, const Base& base)
  {
    out << "Base";
    return out;
  }
 
  class Der : public Base
  {
  };
 
  std::ostream& operator << (std::ostream& out, const Der& der)
  {
    out << "Der";
    return out;
  }
}
 
namespace Good
{
  class Base
  {
  public:
    virtual void Print(std::ostream& out) const
    {
      out << "Base";
    }
  };
 
  std::ostream& operator << (std::ostream& out, const Base& base)
  {
    base.Print(out);
    return out;
  }
 
  class Der : public Base
  {
    virtual void Print(std::ostream& out) const
    {
      out << "Der";
    }
  };
}
 
int main()
{
  {
    // Base ссылается на производный класс (считай винтажный порт.)
    // Но вызывается оператор, который определен для базового класса (считай обычный порт)
    std::cout << "Bad:" << std::endl;
    Bad::Der der;
    Bad::Base* base = &der;
    std::cout << "print: " << *base << std::endl;
  }
 
  {
    // А вот по прежнему вызывается оператор, который определен для базового класса
    // (другого даже нет), но работает все так, как и задумывалось.
    std::cout << "Good:" << std::endl;
    Good::Der der;
    Good::Base* base = &der;
    std::cout << "print: " << *base << std::endl;
  }
 
  return 0;
}
0
kravam
быдлокодер
1700 / 887 / 45
Регистрация: 04.06.2008
Сообщений: 5,494
18.01.2012, 23:15 #17
А на пальцах? Я пойму...
А вообще если челу надо, чтобы переменная имела значение X, надо ей это значение присвоить а не простыни выкладывать.
0
DU
1483 / 1129 / 45
Регистрация: 05.12.2011
Сообщений: 2,279
18.01.2012, 23:22 #18
два раза на пальцах объяснил и один раз на примере, что лучше чем пальцы. Если проблемы с пониманием того, как компилятор подбирает ф-ии для вызова - то тут лучше соответствующий материал поизучать (без обид). Еще раз на пример посмотрите. А лучше откомпилируйте и посмотрите, что выводится.
0
retmas
Жарю без масла
863 / 745 / 168
Регистрация: 13.01.2012
Сообщений: 1,702
18.01.2012, 23:23 #19
Цитата Сообщение от kravam Посмотреть сообщение
Да для винтажного она вызывается, для винтажного. mass [1] указывает на винтажный порт
этого нет и быть не может, потому что operator>> не является виртуальной ф-ей-членом класса Port.
перегрузка ф-ии (а именно она здесь) осуществляется по статическим типам на этапе компиляции. и на этапе же компиляции генерируется код для вызова
C++
1
std::ostream & operator<<(std::ostream & os, const Port & p)
и компилятору нас*ать, каким будет на самом деле динамический тип параметра во время выполнения
0
kravam
быдлокодер
1700 / 887 / 45
Регистрация: 04.06.2008
Сообщений: 5,494
18.01.2012, 23:25 #20
Как можно сделать, я знаю, просто мне кажется мой способ более простой, чем ваш. Давайте поговорим тезисно.

1) По окончанию ввода mass[0] указывает на Port, а mass [1] на элемент типа VintagePort
Согласны?
0
DU
1483 / 1129 / 45
Регистрация: 05.12.2011
Сообщений: 2,279
18.01.2012, 23:36 #21
Уж лучше в таких терминах:
C++
1
2
Port* p1 = new Port();
Port* p2 = new VintagePort();
p1 указывает на порт. p2 указывает на винтажный порт. оба указателя имеют тип Port*.
Что дальше?

Выкладывайте сразу все свои выводы.
0
kravam
быдлокодер
1700 / 887 / 45
Регистрация: 04.06.2008
Сообщений: 5,494
18.01.2012, 23:53 #22
Дальше такой вопрос, в каждом из классов перегружен опреатор << , предположим для простоты , что если вызовется оператор, переопределённый для Port, то выведется, "Port", а если для VintagePort, то "VintagePort". Вопрос: что выведется в этом случае?
C++
1
2
cout << *p1<< endl;        
cout << *p2
Я считаю, что выведет
C++
1
2
Port
VintagePort
, а вы как думаете?

Добавлено через 9 минут
Д и ещё я забыл добавить, VintagePort производный от Port, последний не абстрактный
0
DU
1483 / 1129 / 45
Регистрация: 05.12.2011
Сообщений: 2,279
18.01.2012, 23:54 #23
А я думаю по другому. Вывод зависит от того, как переопределен оператор.
Но если операторы такие:
C++
1
2
3
4
5
6
7
8
9
std::ostream& operator << (std::ostream&, const Port& p)
{
   return out << "Port";
}
 
std::ostream& operator << (std::ostream&, const VintagePort& p)
{
   return out << "VintagePort";
}
То вывод будет не такой, как вы считаете. Вывод будет:
Port
Port

Если сомневаетесь, то еще раз прошу вас скомпилить последный мой пример. Там эта ситуация как раз и заимплеменчена в плохом неймспейсе. В хорошем - хорошая имплементация, у которой таких проблем нет.
0
retmas
Жарю без масла
863 / 745 / 168
Регистрация: 13.01.2012
Сообщений: 1,702
18.01.2012, 23:59 #24
Цитата Сообщение от kravam Посмотреть сообщение
в каждом из классов перегружен опреатор <<
в коде из 1 поста
1) в каждом из классов не перегружен опреатор << и прога делает то, что делает, а не то что вам хочется
2) даже если
Цитата Сообщение от kravam Посмотреть сообщение
предположим для простоты
покажите как вы "перегрузите" << для
C++
1
cout << *p2
0
DU
1483 / 1129 / 45
Регистрация: 05.12.2011
Сообщений: 2,279
19.01.2012, 00:13 #25
И еще для справки:
1.
Нельзя написать виртуальный оператор << (как метод класса) чтобы использование было таким:
C++
1
std::cout << obj;
Можно только в виде свободной ф-ии или захерачить такой оператор в классе std::ostream (а его трогать нельзя, так что такой способ тоже не подходит).

Если вы все же сделаете этот оператор мембером класса, то компилироваться будет только такая запись:
C++
1
obj << std::cout;
Но кто захочет этим пользоваться? Никто. Поэтому используется подход с виртуальными методами печати и одним оператором << для базового класса, в имплементе которого у ссылки, имеющей тип базового класса вызывается виртуальная ф-ия печати.

2. Компилятор подбирает ф-ию на основании статических типов.
Статический тип выражения *p1 - это const Port&. Поэтому всегда привсегда будет вызываться реализация оператора <<, у которого в сигнатуре прописан этот самый const Port&, т.е.
std::ostream& operator << (std::ostream& out, const Port& port);

Если это все еще вас не убедило, то это печально.
0
KeyGen
384 / 291 / 6
Регистрация: 07.08.2011
Сообщений: 790
Записей в блоге: 1
19.01.2012, 00:26  [ТС] #26
DU - реально помог. Спасибо.
0
kravam
быдлокодер
1700 / 887 / 45
Регистрация: 04.06.2008
Сообщений: 5,494
19.01.2012, 01:13 #27
Друзья! Ну, может, я, конечно, в теории слаб, но я хз как ещё объяснять человеку, русским языком спрашиваю: чё должно вывестись? Пишет:
vintage none 45

Хотя выводиться должно было нечто ДРУГОЕ. Мне это другое и надо было. Как ещё объяснять, я не знаю. Может я дурак, может чё.
Ну я быстренько вывел чё ему надо и давай и без того скудные знания под ответ подгонять, каюсь.
0
19.01.2012, 01:13
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
19.01.2012, 01:13
Привет! Вот еще темы с ответами:

Создать базовый класс Car (машина) и производный класс Lorry (грузовик): ООП ошибки - C++
Создать базовый класс Car (машина), характеризуемый торговой маркой (строка), числом цилиндров, мощностью. Определить методы переназначения...

Создать базовый класс Triad и производный класс vector3D - C++
Помогите пожалуйста с заданием. Часть кода(vector3D) сделал, вроде работает Нужно помочь с классом Triad. Вот полное задание: ...

Указатели в базовом классе на базовый класс и производный класс - C++
Пишу контейнер &quot;Бинарное дерево поиска&quot; для частотного словаря. С самим контейнером особо вопросов нету. Вопрос по поводу элементов в...

Наследование: базовый класс Квадрат, производный класс Пирамида - C++
Помогите, пожалуйста!:cry: Добавлено через 6 минут Создать класс КВАДРАТ, член класса- длинна стороны. Предусмотреть в классе методы...


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

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

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