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

Dynamic_cast и полиморфизм - C++

Восстановить пароль Регистрация
 
CHELOVEKPAUK
 Аватар для CHELOVEKPAUK
13 / 3 / 2
Регистрация: 01.08.2013
Сообщений: 332
Записей в блоге: 1
23.06.2014, 10:15     Dynamic_cast и полиморфизм #1
Задача:
В листинге 15.16 после каждого блока try находятся два блока catch, поэтому исключение nbad_index приводит к вызову метода label_val (). Измените программу так, чтобы она содержала один блок catch после каждого блока try и
использовала RTTI для вызова label_val () лишь тогда, когда это необходимо.


Листинг 15.16
Кликните здесь для просмотра всего текста
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
// use_sales.cpp -- вложенные исключения
#include <iostream>
#include "sales.h"
int main()
{
    using std::cout;
    using std::cin;
    using std::endl;
    double vals1[12] =
    {
        1220, 1100, 1122, 2212, 1232, 2334,
        2884, 2393, 3302, 2922, 3002, 3544
    };
    double vals2[12] =
    {
        12, 11, 22, 21, 32, 34,
        28, 29, 33, 29, 32, 35
    };
    Sales sales1(2004, vals1, 12);
    LabeledSales sales2("Blogstar",2005, vals2, 12 );
    cout << "Первый блок try:\n";
    try
    {
        int i;
        cout << "Год = " << sales1.Year() << endl;
        for (i = 0; i < 12; ++i)
        {
            cout << sales1[i] << ' ';
            if (i % 6 == 5)
                cout << endl;
        }
        cout << "Год = " << sales2.Year() << endl;
        cout << "Метка = " << sales2.Label() << endl;
        for (i = 0; i <= 12; ++i)
        {
            cout << sales2[i] << ' ';
            if (i % 6 == 5)
                cout << endl;
        }
        cout << "Конец первого блока try.\n";
    }
    catch(LabeledSales::nbad_index & bad)
    {
        cout << bad.what();
        cout << "Компания: " << bad.label_val() << endl;
        cout << "Неверный индекс: " << bad.bi_val() << endl;
    }
    catch(Sales::bad_index & bad)
    {
        cout << bad.what();
        cout << "Неверный индекс: " << bad.bi_val() << endl;
    }
    cout << "\nСледующий блок try:\n";
    try
    {
        sales2[2] = 37.5;
        sales1[20] = 23345;
        cout << "Конец второго блока try.\n";
    }
    catch(LabeledSales::nbad_index & bad)
    {
        cout << bad.what();
        cout << "Компания: " << bad.label_val() << endl;
        cout << "Неверный индекс: " << bad.bi_val() << endl;
    }
    catch(Sales::bad_index & bad)
    {
        cout << bad.what();
    cout << "Неверный индекс: " << bad.bi_val() << endl;
    }
    cout << "Готово.\n";
    return 0;
}


sales.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
39
40
41
42
43
44
45
46
47
48
49
#include <stdexcept>
#include <cstring>
 
class Sales
{
public:
    enum {MONTHS = 12};   // could be a static const
    class bad_index : public std::logic_error
    {
    private:
        int bi;  // bad index value
    public:
        explicit bad_index(int ix,
            const char * s = "Index error in Sales object\n");
        int bi_val() const {return bi;}
    };
    explicit Sales(int yy = 0);
    Sales(int yy, const double * gr, int n);
    virtual ~Sales() { }
    int Year() const { return year; }
    virtual double operator[](int i) const throw(std::logic_error);
    virtual double & operator[](int i) throw(std::logic_error);
private:
    double gross[MONTHS];
    int year;
};
 
class LabeledSales : public Sales
{
  public:
    static const int STRLEN = 50;  // could be an enum
    class nbad_index : public Sales::bad_index
    {
    private:
        char lbl[STRLEN];
    public:
        nbad_index(const char * lb, int ix,
           const char * s = "Index error in LabeledSales object\n");
        const char * label_val() {return lbl;}
    };
    explicit LabeledSales(const char * lb = "none", int yy = 0);
    LabeledSales(const char * lb, int yy, const double * gr, int n);
    virtual ~LabeledSales() { }
    const char * Label() const {return label;}
    virtual double operator[](int i) const throw(std::logic_error);
    virtual double & operator[](int i) throw(std::logic_error);
private:
    char label[STRLEN];
};


Помогите, не знаю как объединить два блока catch. Думаю нужно сделать один блок Sales::bad_index &, но не знаю как его грамотно расписать. Можно с помощью dynamic_cast проверить является ли тип наследником bad_index, сделать что типа if (проверка на то что dynamic_cast равен 0) и следом расписать алгоритм для производного класса, но если к примеру вместо nbad_index будет совершенно другой класс? как поведет себя программа?
Подскажите пожалуйста.
Лучшие ответы (1)
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
23.06.2014, 10:15     Dynamic_cast и полиморфизм
Посмотрите здесь:

dynamic_cast C++
dynamic_cast C++
dynamic_cast() шаблон C++
Не много не понятно (dynamic_cast) C++
dynamic_cast C++
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Tulosba
:)
Эксперт C++
4377 / 3220 / 297
Регистрация: 19.02.2013
Сообщений: 9,044
23.06.2014, 10:40     Dynamic_cast и полиморфизм #2
Сообщение было отмечено автором темы, экспертом или модератором как ответ
Не очень красиво, но суть такая:
C++
1
2
3
4
5
6
7
8
9
10
11
catch(Sales::bad_index & bad)
{
   try {
      LabeledSales::nbad_index& nbad = dynamic_cast<LabeledSales::nbad_index&>(bad);
      // используем nbad
   }
   catch( std::bad_cast& ) // не удалось выполнить dynamic_cast, т.е. bad не является LabeledSales::nbad_index
   {
       // используем bad как Sales::bad_index
   }
}
CHELOVEKPAUK
 Аватар для CHELOVEKPAUK
13 / 3 / 2
Регистрация: 01.08.2013
Сообщений: 332
Записей в блоге: 1
23.06.2014, 11:31  [ТС]     Dynamic_cast и полиморфизм #3
Цитата Сообщение от Tulosba Посмотреть сообщение
Не очень красиво, но суть такая:
Tulosba, т.е. если не получается привести к nbad_index, то вылетает исключение, которое мы перехватываем, и обрабатываем объект как bad_index?
Tulosba
:)
Эксперт C++
4377 / 3220 / 297
Регистрация: 19.02.2013
Сообщений: 9,044
23.06.2014, 11:39     Dynamic_cast и полиморфизм #4
Цитата Сообщение от CHELOVEKPAUK Посмотреть сообщение
т.е. если не получается привести к nbad_index, то вылетает исключение, которое мы перехватываем, и обрабатываем объект как bad_index?
Именно так. Если бы вместо ссылки аргументом dynamic_cast был бы указатель, то вместо исключения получили бы нулевой указатель (nullptr).
Voivoid
 Аватар для Voivoid
580 / 256 / 12
Регистрация: 31.03.2013
Сообщений: 1,283
23.06.2014, 12:05     Dynamic_cast и полиморфизм #5
Задача просто жесть. Никогда не надо так делать
Tulosba
:)
Эксперт C++
4377 / 3220 / 297
Регистрация: 19.02.2013
Сообщений: 9,044
23.06.2014, 12:07     Dynamic_cast и полиморфизм #6
Цитата Сообщение от Voivoid Посмотреть сообщение
Задача просто жесть. Никогда не надо так делать
Использование RTTI вообще в большинстве случаев показатель кривого дизайна.
Yandex
Объявления
23.06.2014, 12:07     Dynamic_cast и полиморфизм
Ответ Создать тему
Опции темы

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