Форум программистов, компьютерный форум, киберфорум
C++
Войти
Регистрация
Восстановить пароль
 
Рейтинг 4.75/4: Рейтинг темы: голосов - 4, средняя оценка - 4.75
1 / 1 / 2
Регистрация: 18.05.2017
Сообщений: 8
1

Множественное наследование

18.05.2017, 19:21. Просмотров 772. Ответов 3
Метки нет (Все метки)

Добрый день, коллективный разум, прошу твоей помощи.
Вопрос в следующем: как имея следующую структуру классов использующих множественное наследование реализовать задумку.
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
template <class P>
class Basic {
protected:
  typedef typename P::ValueType ValueType;
  typedef typename P::PointerType PointerType;
public:
  inline operator ValueType() {
    return ((P *)this)->operator*();
  }
  
  inline operator PointerType() {
    return ((P *)this)->operator->();
  }
};
 
template <class P>
class VoidPtr {
public:
  inline void * ToVoidPtr() {
    return (void *)(P *)(this)->operator->();
  }
  
  inline operator void *() {
    return ToVoidPtr();
  }
};
 
template <class T, template <class> class ... Conversation>
class Pointer : public Conversation<Pointer<T>> ... {
public:
  typedef T ValueType;
  typedef ValueType * PointerType;
protected:
  void * data_;
public:
  Pointer() { data_ = nullptr; }
  Pointer(T& data) { data_ = (void *)&data; }
  T& operator*() {
    return *(T *)data_;
  }
  T * operator->() {
    return (T *)data_;
  }
};
Если создать класс Pointer следующим образом:
C++
1
Pointer<int>
все замечательно. Если унаследоваться только от одного класса
C++
1
Pointer<int, Basic>
- все ок. Все работает, преобразование к базовому типу тоже ок. Но если указать
C++
1
Pointer<int, Basic, VoidPtr>
, то при попытке преобразовать к void * или к базовому типу ошибка доступа к памяти из-за того что указатель в классе Basic классе Pointer указывают на разные участки памяти (в классе Basic или VoidPtr указатель на класс Pointer сдвинут на несколько байт вперед) и ну Вы поняли. Как можно выйти из данной ситуации.
0
Programming
Эксперт
94731 / 64177 / 26122
Регистрация: 12.04.2006
Сообщений: 116,782
18.05.2017, 19:21
Ответы с готовыми решениями:

Множественное наследование - так ли это плохо?
Вопрос не в том, что это такое и пр. С момента изучения Си++ во всех книжках все авторы (кроме...

Множественное наследование от Form
class A { }; class B { }; class C : public Form,public A,public B {

Множественное виртуальное наследование от класса TForm
У меня есть некий базовый класс TWindowBase (производный от TForm), который я наследую во все...

Множественное наследование VCL& экспорт TThread
Здравствуйте. 1) Кто-нибудь пробовал экспортировать переменные типа TThread? extern TNewThread...

3
зомбяк
1420 / 1084 / 310
Регистрация: 14.05.2017
Сообщений: 3,551
18.05.2017, 21:29 2
наверно, сделать
C++
1
Pointer<VoidPtr, int, Basic>
0
1 / 1 / 2
Регистрация: 18.05.2017
Сообщений: 8
19.05.2017, 07:26  [ТС] 3
Первый параметр в классе Pointer это тип сохраняемый в указателе, все последующие необязательные параметры - полити преобразования. Их может быть от 0 до очень много. Тоесть
C++
1
Pointer<int>
указатель на int
C++
1
Pointer<int, Basic>
указатель на int но с возможность преобразованния к базовому типу
C++
1
Pointer<int, DoubleConvert, VoidPtr, Basic>
возможно преобразовать к void *, базовому типу и к типу double, например
Тоесть стратегий преобразования может быть очень много и они задаются при определении шаблона.
0
1 / 1 / 2
Регистрация: 18.05.2017
Сообщений: 8
21.05.2017, 12:01  [ТС] 4
Нашел решение, может быть можно реализовать и как-то по другому.
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
namespace ConversationStrategy {
    template <class P>
    class Strategy {
    public:
        void * data_;
    };
 
    template <class P>
    class None : virtual public Strategy<P> {};
 
    template <class P>
    class Basic : virtual public Strategy<P> {
    protected:
        typedef typename P::ValueType ValueType;
        typedef typename P::PointerType PointerType;
        typedef typename P::ClassType ClassType;
    public:
        inline operator ValueType() {
            return *(PointerType)Strategy<ClassType>::data_;
        }
 
        inline operator PointerType() {
            return (PointerType)Strategy<ClassType>::data_;
        }
    };
 
    template <class P>
    class VoidPtr : virtual public Strategy<P> {
    protected:
        typedef typename P::ClassType ClassType;
    public:
        inline void * ToVoidPtr() {
            return Strategy<ClassType>::data_;
        }
 
        inline operator void *() {
            return ToVoidPtr();
        }
    };
}
 
template <class T, template <class> class ... Conversation>
class Pointer : virtual private ConversationStrategy::Strategy<Pointer<T>>, virtual public Conversation<Pointer<T>> ... {
public:
    typedef T ValueType;
    typedef ValueType * PointerType;
    typedef ValueType * const ConstValueType;
    typedef ValueType& ReferenceType;
    typedef const ValueType& ConstReferenceType;
    typedef Pointer<ValueType> ClassType;
public:
    Pointer() { }
 
    Pointer(T& data) {
        ConversationStrategy::Strategy<ClassType>::data_ = (void *)&data;
    }
 
    T& operator*() {
        return *(T *)ConversationStrategy::Strategy<ClassType>::data_;
    }
 
    T * operator->() {
        return (T *)ConversationStrategy::Strategy<ClassType>::data_;
    }
};
Суть заключается в создании базового класса Strategy от которого и будут далее наследоваться все политики преобразований. В нем находится открытый член void * data_. Далее политика преобразования Basic, например, виртуально наследуется от Strategy и получает доступ к data_. Так как наследование виртуальное, то все политики получат один и тот же базовый класс Strategy и, следовательно, один член data_.
Класс Pointer, также виртуально наследуется от Strategy и от политик преобразования. Наследование от Strategy нужно так как, политик преобразования может и не быть вовсе.
В итоге все классы в дереве наследований имеют только одну и ту же копию Strategy и один член data_.
1
IT_Exp
Эксперт
87844 / 49110 / 22898
Регистрация: 17.06.2006
Сообщений: 92,604
21.05.2017, 12:01

Заказываю контрольные, курсовые, дипломные и любые другие студенческие работы здесь.

Наследование шаблоном шаблона (наследование конструктора)
Всем привет! Думаю, что эта тема хоть и касается общих вопросов программирования, но будет уместна...

Множественное редактирование в DBGrid
Объясните, пожалуйста, начинающему програмисту на С++, как осуществить множественное редактирование...

Множественное наследование
Создать иерархии наследования: телевизор, цифровое устройство - монитор. Создать динамический...

Множественное наследование POSTAVKA
Разработайте базовый класс POSTAVKA. Элемент класса: -название фирмы-поставщика; -цена...


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

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

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