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

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

Восстановить пароль Регистрация
 
 
Рейтинг: Рейтинг темы: голосов - 14, средняя оценка - 4.86
KeyGen
 Аватар для KeyGen
333 / 289 / 6
Регистрация: 07.08.2011
Сообщений: 789
Записей в блоге: 1
18.01.2012, 20:35     Производный класс. Перегрузка << (cout). #1
При инициализации массива указателей базового класса при вызове 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;
}
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
DU
1477 / 1053 / 45
Регистрация: 05.12.2011
Сообщений: 2,279
18.01.2012, 23:36     Производный класс. Перегрузка << (cout). #21
Уж лучше в таких терминах:
C++
1
2
Port* p1 = new Port();
Port* p2 = new VintagePort();
p1 указывает на порт. p2 указывает на винтажный порт. оба указателя имеют тип Port*.
Что дальше?

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

Добавлено через 9 минут
Д и ещё я забыл добавить, VintagePort производный от Port, последний не абстрактный
DU
1477 / 1053 / 45
Регистрация: 05.12.2011
Сообщений: 2,279
18.01.2012, 23:54     Производный класс. Перегрузка << (cout). #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

Если сомневаетесь, то еще раз прошу вас скомпилить последный мой пример. Там эта ситуация как раз и заимплеменчена в плохом неймспейсе. В хорошем - хорошая имплементация, у которой таких проблем нет.
retmas
Жарю без масла
803 / 685 / 143
Регистрация: 13.01.2012
Сообщений: 1,580
18.01.2012, 23:59     Производный класс. Перегрузка << (cout). #24
Цитата Сообщение от kravam Посмотреть сообщение
в каждом из классов перегружен опреатор <<
в коде из 1 поста
1) в каждом из классов не перегружен опреатор << и прога делает то, что делает, а не то что вам хочется
2) даже если
Цитата Сообщение от kravam Посмотреть сообщение
предположим для простоты
покажите как вы "перегрузите" << для
C++
1
cout << *p2
DU
1477 / 1053 / 45
Регистрация: 05.12.2011
Сообщений: 2,279
19.01.2012, 00:13     Производный класс. Перегрузка << (cout). #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);

Если это все еще вас не убедило, то это печально.
KeyGen
 Аватар для KeyGen
333 / 289 / 6
Регистрация: 07.08.2011
Сообщений: 789
Записей в блоге: 1
19.01.2012, 00:26  [ТС]     Производный класс. Перегрузка << (cout). #26
DU - реально помог. Спасибо.
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
19.01.2012, 01:13     Производный класс. Перегрузка << (cout).
Еще ссылки по теме:

C++ Перегрузка cout
C++ Разработать производный от абстрактного класса Figure класс и класс, производный от производного
C++ Написать класс строка и производный класс, ее шифрующий

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

Или воспользуйтесь поиском по форуму:
kravam
быдлокодер
 Аватар для kravam
1512 / 872 / 44
Регистрация: 04.06.2008
Сообщений: 5,270
19.01.2012, 01:13     Производный класс. Перегрузка << (cout). #27
Друзья! Ну, может, я, конечно, в теории слаб, но я хз как ещё объяснять человеку, русским языком спрашиваю: чё должно вывестись? Пишет:
vintage none 45

Хотя выводиться должно было нечто ДРУГОЕ. Мне это другое и надо было. Как ещё объяснять, я не знаю. Может я дурак, может чё.
Ну я быстренько вывел чё ему надо и давай и без того скудные знания под ответ подгонять, каюсь.
Yandex
Объявления
19.01.2012, 01:13     Производный класс. Перегрузка << (cout).
Ответ Создать тему
Опции темы

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