С Новым годом! Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Карта форума Темы раздела Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.78/9: Рейтинг темы: голосов - 9, средняя оценка - 4.78
13 / 3 / 2
Регистрация: 01.08.2013
Сообщений: 332
Записей в блоге: 1
1

Dynamic_cast и полиморфизм

23.06.2014, 10:15. Показов 1730. Ответов 5
Метки нет (Все метки)

Author24 — интернет-сервис помощи студентам
Задача:
В листинге 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 будет совершенно другой класс? как поведет себя программа?
Подскажите пожалуйста.
0
Лучшие ответы (1)
IT_Exp
Эксперт
8794 / 1073 / 104
Регистрация: 17.06.2006
Сообщений: 12,602
Блог
23.06.2014, 10:15
Ответы с готовыми решениями:

Dynamic_cast
#include &lt;iostream&gt; #define STOP cin.get(); using std::cin; using std::cout; class A {...

dynamic_cast
Подскажите пожалуйста, возникла проблема. Есть базовый клас CStep. И пока один наследник: class...

dynamic_cast
#include &lt;iostream&gt; class B { public: virtual void foo(){std::cout&lt;&lt;&quot;B&quot;;} private: ...

dynamic_cast
Уважаемые, подскажите теорию. Как устроен данный каст? Не нашел его определений. В студии он зашит...

5
:)
Эксперт С++
4773 / 3267 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
23.06.2014, 10:40 2
Лучший ответ Сообщение было отмечено SatanaXIII как решение

Решение

Не очень красиво, но суть такая:
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
   }
}
1
13 / 3 / 2
Регистрация: 01.08.2013
Сообщений: 332
Записей в блоге: 1
23.06.2014, 11:31  [ТС] 3
Цитата Сообщение от Tulosba Посмотреть сообщение
Не очень красиво, но суть такая:
Tulosba, т.е. если не получается привести к nbad_index, то вылетает исключение, которое мы перехватываем, и обрабатываем объект как bad_index?
0
:)
Эксперт С++
4773 / 3267 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
23.06.2014, 11:39 4
Цитата Сообщение от CHELOVEKPAUK Посмотреть сообщение
т.е. если не получается привести к nbad_index, то вылетает исключение, которое мы перехватываем, и обрабатываем объект как bad_index?
Именно так. Если бы вместо ссылки аргументом dynamic_cast был бы указатель, то вместо исключения получили бы нулевой указатель (nullptr).
1
710 / 283 / 16
Регистрация: 31.03.2013
Сообщений: 1,340
23.06.2014, 12:05 5
Задача просто жесть. Никогда не надо так делать
0
:)
Эксперт С++
4773 / 3267 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
23.06.2014, 12:07 6
Цитата Сообщение от Voivoid Посмотреть сообщение
Задача просто жесть. Никогда не надо так делать
Использование RTTI вообще в большинстве случаев показатель кривого дизайна.
0
23.06.2014, 12:07
BasicMan
Эксперт
19315 / 2622 / 84
Регистрация: 17.02.2009
Сообщений: 10,364
Блог
23.06.2014, 12:07
Помогаю со студенческими работами здесь

dynamic_cast
часть кода такая list&lt;good *&gt;::const_iterator it = p.exp.begin(), end = p.exp.end(); for (; it...

dynamic_cast
Помогите разобраться с dynamic_cast, а точнее с вот этой записью void f() { A* pa = new A; ...

Проблема с dynamic_cast
Есть такой простенький делегат, писавшийся и тестировавшийся под mingw: Код Delegate.h #pragma...

Dynamic_cast реализация
Добрый вечер, есть застоявшаясь фраза: &quot;dynamic_cast&lt;Type*&gt;( pointer ) очень медленное, его...


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

Или воспользуйтесь поиском по форуму:
6
Ответ Создать тему
Блоги программистов
Обновление сайта www.historian.b­y
Reglage 05.01.2025
Обещал подвести итоги 2024 года для сайта. Однако начну с того, что изменилось за неделю. Добавил краткий урок по последовательности действий при анализе вредоносных файлов и значительно улучшил урок. . .
Как использовать GraphQL в C# с HotChocolate
Programming 05.01.2025
GraphQL — это современный подход к разработке API, который позволяет клиентам запрашивать только те данные, которые им необходимы. Это делает взаимодействие с API более гибким и эффективным по. . .
Модель полного двоичного суматора с помощью логических операций (python)
AlexSky-coder 04.01.2025
def binSum(x:list, y:list): s=^y] p=x and y for i in range(1,len(x)): s. append((x^y)^p) p=(x and y)or(p and (x or y)) return s x=list() y=list()
Это мы не проходили, это нам не задавали...(аси­­хронный счётчик с управляющим сигналом задержки).
Hrethgir 04.01.2025
Асинхронный счётчик на сумматорах (шестиразрядный по числу диодов на плате, но наверное разрядов будет больше - восемь или шестнадцать, а диоды на старшие), так как триггеры прошли тестирование и. . .
Руководство по созданию бота для Телеграм на Python
IT_Exp 04.01.2025
Боты для Телеграм представляют собой автоматизированные программы, которые выполняют различные задачи, взаимодействуя с пользователями через интерфейс мессенджера. В данной статье мы рассмотрим,. . .
Применение компонентов PrimeVue в Vue.js 3 на TypeScript
BasicMan 04.01.2025
Введение в PrimeVue и настройка окружения PrimeVue представляет собой мощную библиотеку компонентов пользовательского интерфейса для Vue. js 3, которая предоставляет разработчикам богатый набор. . .
Как стать Senior developer
cpp_developer 04.01.2025
В современной индустрии разработки программного обеспечения позиция Senior Developer представляет собой не просто следующую ступень карьерной лестницы, а качественно новый уровень профессионального. . .
Что известно о дате выхода Windows 12 и чего от нее ждать
IT_Exp 04.01.2025
В мире технологий постоянно происходят изменения, и операционные системы не являются исключением. Windows 11, выпущенная в октябре 2021 года, принесла множество инноваций и улучшений, но. . .
Что новенького в .NET Core 9
Programming 04.01.2025
Обзор ключевых изменений в . NET Core 9 Платформа . NET Core продолжает активно развиваться, и версия 9 представляет собой значительный шаг вперед в эволюции этой технологии. Новый релиз. . .
Инструкция по установке python3.13.1 в Debian 12
AlexSky-coder 03.01.2025
sudo apt update sudo apt install build-essential zlib1g-dev libncurses5-dev libgdbm-dev libnss3-dev libssl-dev libreadline-dev libffi-dev wget. . .
Затестил триггеры. архив проекта прилагаю с GOA файлами в настройках архиватора проектов.
Hrethgir 03.01.2025
В этот раз нет закольцованности, потому что от неё только глюки, как я понял, логика не вырезанная. Триггеры очень быстрые если верить измерениям с помощью анализатора от Gowin. Есть ещё регистры,. . .
Python в помощь DevOps
IT_Exp 03.01.2025
Причины использования Python в работе DevOps Python стал неотъемлемой частью мира DevOps, и это не случайно. Этот язык программирования обладает множеством преимуществ, которые делают его. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru