Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.76/41: Рейтинг темы: голосов - 41, средняя оценка - 4.76
1090 / 588 / 121
Регистрация: 11.11.2008
Сообщений: 1,544

Вектор объектов разных классов. Доступ к свойствам элемента вектора

28.05.2013, 09:11. Показов 8341. Ответов 31
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
есть базовый класс
C++
1
2
3
4
class A
{
//...
}
есть производные классы
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
class A1 : public A
{
//...
public:
int a;
}
 
class A2 : public A
{
//...
public:
int a;
}
создаю вектор объектов
C++
1
2
3
4
std::vector<A> v;
v.push_back(new A1());
v.push_back(new A2());
//и т.д.
как теперь получать доступ к a, которого нет в базовом классе?
насколько приемлемо писать так:
C++
1
2
3
4
5
6
7
8
if(dynamic_cast<A1 *>(v[i]))
        {
        a_get=((A1 *)(v[i]))->a;
        }
if(dynamic_cast<A2 *>(v[i]))
        {
        a_get=((A2 *)(v[i]))->a;
        }
особенно если производных классов будет много?
0
Лучшие ответы (1)
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
28.05.2013, 09:11
Ответы с готовыми решениями:

Вектор объектов разных классов. Полиморфим
В общем проблема такая: есть вектор объектов разных классов, надо вывести информацию о каждом объекте. Класс А абстрактный; В потомок А;...

Доступ к свойствам объектов
Имеется метод, который возвращает несколько объектов вложенных друг в друга. Например в таком виде (при вызове print_r): Array ( ...

Получить доступ к свойствам объектов из базы
У меня есть таблица, в которой записаны объекты, содержащие некоторые свойства. Я указал для dropdownlist, что в качестве содержимого, хочу...

31
503 / 352 / 94
Регистрация: 22.03.2011
Сообщений: 1,112
30.05.2013, 09:35
Студворк — интернет-сервис помощи студентам
ITcrusader, я понимаю), но как я и ответил ранее, более конкретный ответ кроется за применением.
0
Почетный модератор
Эксперт С++
 Аватар для SatanaXIII
5851 / 2862 / 392
Регистрация: 01.11.2011
Сообщений: 6,906
30.05.2013, 09:38
Цитата Сообщение от AntonChik Посмотреть сообщение
насколько приемлемо писать так:
AntonChik, а чем вас не устраивает ваше же статическое приведение типа? Работает.
C++
1
2
3
4
5
6
vector<A*> v;
v.push_back(new A1());
v.push_back(new A2());
 
((A1*)v[0])->a = 99;
((A2*)v[1])->a = 200;
Только верно ли будет в один вектор пихать разные объекты? Если у нас в А1 будет много полей, а в А2 ни одного, справится ли с этим vector?
И на сколько аккуратным надо быть, если таких унаследованных классов будет много, чтобы не поступить вот так:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
class A1: public A
{public:
 
int a;}; // !!!
 
class A2: public A
{public:
 
float a;}; // !!!
 
 
vector<A*> v;
v.push_back(new A1());
v.push_back(new A2());
 
((A1*)v[0])->a = 99;
((A2*)v[1])->a = 200;
 
((A1*)v[1])->a == ?;
((A2*)v[0])->a == ?;
0
Эксперт С++
 Аватар для ITcrusader
179 / 165 / 21
Регистрация: 12.02.2013
Сообщений: 410
30.05.2013, 09:41
да вообще, идея применения приведения вниз по иерархии - в большинстве случаев - плоха, поэтому, автор, пости реальный пример с подробными разъяснениями)))
0
Почетный модератор
Эксперт С++
 Аватар для SatanaXIII
5851 / 2862 / 392
Регистрация: 01.11.2011
Сообщений: 6,906
30.05.2013, 09:45
Цитата Сообщение от ITcrusader Посмотреть сообщение
идея применения приведения вниз по иерархии - в большинстве случаев - плоха
Чем? Поясните пожалуйста. Вон на том же конкретном маленьком примере. И как сделать лучше/по другому?
0
1090 / 588 / 121
Регистрация: 11.11.2008
Сообщений: 1,544
30.05.2013, 09:53  [ТС]
Цитата Сообщение от SatanaXIII Посмотреть сообщение
а чем вас не устраивает ваше же статическое приведение типа? Работает.
не прокатит перебирая элементы вектора в цикле.

Цитата Сообщение от SatanaXIII Посмотреть сообщение
Только верно ли будет в один вектор пихать разные объекты?
верно, если есть необходимость.
Цитата Сообщение от SatanaXIII Посмотреть сообщение
Если у нас в А1 будет много полей, а в А2 ни одного, справится ли с этим vector?
справится.
Цитата Сообщение от SatanaXIII Посмотреть сообщение
И на сколько аккуратным надо быть...
предельно аккуратным надо быть всегда)

Цитата Сообщение от ITcrusader Посмотреть сообщение
поэтому, автор, пости реальный пример с подробными разъяснениями)))
этот пример вполне сойдет за реальный, ни больше ни меньше.
чувствую, что наверное все-таки придется расширить базовый(промежуточно базовый) класс или поменять логику, но все же сам момент интересен.
0
Почетный модератор
Эксперт С++
 Аватар для SatanaXIII
5851 / 2862 / 392
Регистрация: 01.11.2011
Сообщений: 6,906
30.05.2013, 09:56
Цитата Сообщение от AntonChik Посмотреть сообщение
не прокатит перебирая элементы вектора в цикле.
Для этого есть итераторы. И раз контейнер
Цитата Сообщение от AntonChik Посмотреть сообщение
справится
, то и итератор должен.
0
503 / 352 / 94
Регистрация: 22.03.2011
Сообщений: 1,112
30.05.2013, 09:57
Я понял вашу логику)). Дайте определение массива (в данном случае вектора) и Вы поймете, что Вам делать.
0
1090 / 588 / 121
Регистрация: 11.11.2008
Сообщений: 1,544
30.05.2013, 09:58  [ТС]
и как итератор поможет в определении класса элемента?)
0
Эксперт С++
 Аватар для ITcrusader
179 / 165 / 21
Регистрация: 12.02.2013
Сообщений: 410
30.05.2013, 09:58
Лучший ответ Сообщение было отмечено как решение

Решение

SatanaXIII, блин. Я устал уже тут писать про расширение иерархии. Не всегда красиво смотреться будет, конечно, зависит от конкретной области моделирования. Но освобождает от необходимости приведения типов, проверки правильности приведения и т.д.
Далее, все зависит от того еще, например, просто ли мы хотим выполнить GetProperty. Или, например, запустить операцию изменения состояния объекта, которой нет в базовом, но которая есть в паре производных, а в остальных производных на том же уровне отсутствует, тут можно не расширять иерархию, но реализовать эту функцию везде, просто в тех классах, где её быть не должно - оставить её пустой, холостой, как угодно. Или выбрасывающей обрабатываемое исключение, или, или...

Ясно лишь, что если возникает потребность в подобного рода приведении - это наводит на мысль о том, что спроектировать лучше по-другому. Бред - использовать наследование, фишка которого - обеспечить общность, а при этом сидеть и от общего переходить к частному, обрабатывая разные вариации. Думаю, будет правильно сказать, что наследование - призвано обобщить нечто общее - здесь все наоборот.
3
Почетный модератор
Эксперт С++
 Аватар для SatanaXIII
5851 / 2862 / 392
Регистрация: 01.11.2011
Сообщений: 6,906
30.05.2013, 10:05
Цитата Сообщение от AntonChik Посмотреть сообщение
и как итератор поможет в определении класса элемента?)
Никак. Но по крайней мере мы не обратимся в середину какого-то элемента. Я об этом.
В целом задумка как-то не очень по моему - разнотипные объекты пихать в одну кучу.

Цитата Сообщение от ITcrusader Посмотреть сообщение
Я устал уже тут писать про расширение иерархии.
Выложите куда-нибудь в блог или ссылку приводите на самый развернутый ответ, чтобы по десять раз одно и то же не повторять. )
0
1090 / 588 / 121
Регистрация: 11.11.2008
Сообщений: 1,544
30.05.2013, 11:05  [ТС]
ITcrusader, спасибо огромное. Я получил абсолютно исчерпывающий ответ, какой и искал. Замучу вариант с вирт. функциями, как и подозревал.
Отдельное спасибо за книжку, когда-то мельком листал ее, а сейчас понимаю что пришло время читануть ее глубоко и вдумчиво)
0
Комп_Оратор)
Эксперт по математике/физике
 Аватар для IGPIGP
9007 / 4708 / 630
Регистрация: 04.12.2011
Сообщений: 14,003
Записей в блоге: 16
30.05.2013, 21:28
Полиморфность при наследовании определяется наличием виртуальных методов. Вот пример не полиморфного наследования. Ни лобовое применения typeinfo ни косвенное выяснение типов через указатель базового класса не различают наследников на которые он указывает. Доступа к данным наследников (легального)) по такому указателю нет:
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
//если в базовом классе нет виртуальных методов, то наследование не полиморфно
//и типом указателя базового класса всегда является базовый класс, независимо от
//инициализации, 
#include <iostream>
#include <typeinfo.h>
using namespace std;
class A
{
public:
int a;
A():a(0){}
A(int _a):a(_a){}
};
class B: public A
{
public:
int b;
B():A(),b(0){}
B(int _a, int _b):A(_a), b(_b){}
};
class C: public A
{
public:
int c;
C():A(),c(0){}
C(int _a, int _c):A(_a),c(_c){}
};
template <class T>
void showType(T *a){
cout<<typeid(*a).name()<<endl;
}
int main() {
A* ptrBase;
A a(10);
B b(1,2);
C c(3,4);
ptrBase=&a;
showType(ptrBase);
ptrBase=&b;
showType(ptrBase);
//cout<<ptrBase->a<<endl;//это нормально, что и не удивительно
//ptrBase->b; //это даже не компилирутся
//cout<<(dynamic_cast<B*>(ptrBase))->b<<endl;//это тоже не компилирутся: dynamic_cast: "A" не является полиморфным типом
cout<<((B*)(ptrBase))->b<<endl;// а напролом - работает, но врядли это нормально))
ptrBase=&c;
showType(ptrBase);
system("pause");
cout << endl;
return 0;
}
А вот классы-наследники переопределяющие метод виртуальный базового класса. Причем сами методы не используются. Они определены чтобы сделать наследование полиморфным. И наследники стали полиморфны что и видно в коде. Теперь можно выяснить их тип и получить доступ к их данным (не методам) через указатель на базовый класс вполне легально:
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
#include <iostream>
#include <typeinfo.h>
using namespace std;
class A
{
public:
int a;
A():a(0){}
A(int _a):a(_a){}
virtual void Who_s_there_You_Horatio(){cout<<"A"<<endl;}
};
class B: public A
{
public:
int a;
int b;
B():A(),b(0){}
B(int _a, int _b):A(_a),b(_b){}
void Who_s_there_You_Horatio(){cout<<"B"<<endl;}
};
class C: public A
{
public:
int a;
int c;
C():A(),c(0){}
C(int _a, int _c):A(_a),c(_c){}
void Who_s_there_You_Horatio(){cout<<"C"<<endl;}
};
template <class T>
void showType(T *a){
cout<<typeid(*a).name()<<endl;
}
//мы не использовали виртуальные методы, но важно, что они есть
//это позволяет различить типы по указателю и получить доступ и к данным наследников
//практически без бандитизма))
int main() {
A* ptrBase;
A a(10);
B b(1,2);
C c(3,4);
 
ptrBase=&a;
showType(ptrBase);
ptrBase=&b;
showType(ptrBase);
cout<<((B*)(ptrBase))->b<<endl; //
ptrBase=&c;
showType(ptrBase);
system("pause");
cout << endl;
return 0;
}
хотя и методы тоже вызвать можно.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
30.05.2013, 21:28
Помогаю со студенческими работами здесь

дан вектор А (N). написать программу нахождения минимального элемента вектора и его индекса. если индекс элемента парный, то часть вектора перед миним
дан вектор А (N). написать программу нахождения минимального элемента вектора и его индекса. если индекс элемента парный, то часть вектора...

Доступ к полям и свойствам объектов которые лежат в ArrayList
Всем привет! Есть класс Apple с методом getWeight. Объекты этого класса лежат в ArrayList. Как получить доступ к методу объекта который...

Разработать иерархию классов, демонстрирующее работу с коллекцией объектов разных классов
Задание: Разработать в соответствии с индивидуальным заданием иерархию классов и приложение, демонстрирующее работу с коллекцией объектов...

Доступ к свойствам пользовательского элемента управления
Создал пользовательский элемент управления.Нарисовал на нём круг и линию с помощью Visual Basic PowerPacks. Как осуществить доступ к нему...

Заданный вектор А, состоящий из n элементов. Переставить компоненты вектора, размещенные после самого элемента вектора А
Заданный вектор А, состоящий из n элементов. Переставить компоненты вектора, размещенные после самого элемента вектора А на начало, а...


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

Или воспользуйтесь поиском по форуму:
32
Ответ Создать тему
Новые блоги и статьи
Программный контроль заполнения реквизита табличной части документа
Maks 02.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "СписаниеМатериалов", разработанного в конфигурации КА2. Задача: реализовать контроль заполнения реквизита табличной части. . .
wmic не является внутренней или внешней командой
Maks 02.04.2026
Решение: DISM / Online / Add-Capability / CapabilityName:WMIC~~~~ Отсюда: https:/ / winitpro. ru/ index. php/ 2025/ 02/ 14/ komanda-wmic-ne-naydena/
Программная установка даты и запрет ее изменения
Maks 02.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "СписаниеМатериалов", разработанного в конфигурации КА2. Задача: при создании документов установить период списания автоматически. . .
Вывод данных в справочнике через динамический список
Maks 01.04.2026
Реализация из решения ниже выполнена на примере нетипового справочника "Спецтехника" разработанного в конфигурации КА2. Задача: вывести данные из ТЧ нетипового документа. . .
Функция заполнения текстового поля в реквизите формы документа
Maks 01.04.2026
Алгоритм из решения ниже реализован на нетиповом документе "ВыдачаОборудованияНаСпецтехнику" разработанного в конфигурации КА2, в дополнении к предыдущему решению. На форме документа создается. . .
К слову об оптимизации
kumehtar 01.04.2026
Вспоминаю начало 2000-х, университет, когда я писал на Delphi. Тогда среди программистов на форумах активно обсуждали аккуратную работу с памятью: нужно было следить за переменными, вовремя. . .
Идея фильтра интернета (сервер = слой+фильтр).
Hrethgir 31.03.2026
Суть идеи заключается в том, чтобы запустить свой сервер, о чём я если честно мечтал давно и давно приобрёл книгу как это сделать. Но не было причин его запускать. Очумелые учёные напечатали на. . .
Модель здравосоХранения 6. ESG-повестка и устойчивое развитие; углублённый анализ кадрового бренда
anaschu 31.03.2026
В прикрепленном документе раздумья о том, как можно поменять модель в будущем
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru