У меня есть код, который выглядит следующим образом:
template<typename T>
struct memory_block {
// Very not copiable, this class cannot move
memory_block(memory_block const&) = delete;
memory_block(memory_block const&&) = delete;
memory_block(memory_block&) = delete;
memory_block(memory_block&&) = delete;
memory_block& operator=(memory_block const&) = delete;
memory_block& operator=(memory_block&&) = delete;
// The only constructor construct the `data` member with args
template<typename... Args>
explicit memory_block(Args&&... args) noexcept :
data{std::forward<Args>(args)...} {}
T data;
};
template<typename T>
struct special_block : memory_block<T> {
using memory_block<T>::memory_block;
std::vector<double> special_data;
};
// There is no other inheritance. The hierarchy ends here.
Теперь я должен сохранить эти типы в стертом хранилище. Я выбрал вектор void*
в качестве контейнера. Я вставляю указатели члена data
в вектор:
struct NonTrivial { virtual ~NonTrivial() {} };
// exposed to other code
std::vector<void*> vec;
// My code use dynamic memory instead of static
// Data, but it's simpler to show it that way.
static memory_block<int> data0;
static special_block<NonTrivial> data1;
void add_stuff_into_vec() {
// Add pointer to `data` member to the vector.
vec.emplace_back(&(data0->data));
vec.emplace_back(&(data1->data));
}
Затем в коде я получаю доступ к данным:
// Yay everything is fine, I cast the void* to it original type
int* data1 = static_cast<int*>(vec[0]);
NonTrivial* data1 = static_cast<NonTrivial*>(vec[1]);
Проблема в том, что я хочу получить доступ к special_data
в нетривиальном случае:
// Pretty sure this cast is valid! (famous last words)
std::vector<double>* special = static_cast<special_block<NonTrivial>*>(
static_cast<memory_block<NonTrivial>*>(vec[1]) // (1)
);
Итак, вопрос
Проблема возникает в строке (1)
: у меня есть указатель на data
(типа NonTrivial
), который является членом memory_block<NonTrivial>
. Я знаю, что void*
всегда будет указывать на первый элемент данных memory_block<T>
.
Значит, приведение void*
к первому члену класса в сейф класса? Если нет, есть ли другой способ сделать это? Если это может сделать все проще, я могу избавиться от наследства.
Кроме того, у меня нет проблем с использованием std::aligned_storage
в этом случае. Если это решит проблему, я воспользуюсь этим.
Я надеялся, что стандартная раскладка поможет мне в этом случае, но мое статическое утверждение, похоже, не работает.
Мое статическое утверждение:
static_assert(
std::is_standard_layout<special_block<NonTrivial>>::value,
"Not standard layout don't assume anything about the layout"
);