ПроблемаНовый не перегруженный оператор для возможно переравнивающегося класса возникает, когда вы не перегружаете оператор должным образом new/new[]
и вы используете этот оператор, чтобы создать объект с требованием выравнивания, заданным с alignas
. Шашка поднимает дефект для этих версий оператора метания и без метания new/new[]
.
void* operator new(std::size_t size)
void* operator new(std::size_t size, const std::nothrow_t&)
void* operator new[](std::size_t size)
void* operator new[](std::size_t size, const std::nothrow_t&)
Использование alignas
указывает, что оператор по умолчанию не ожидается new/new[]
чтобы удовлетворить требованию выравнивания или объекту, и что объект может быть выровнен. Тип переровнен, если вы используете alignas
чтобы сделать требование выравнивания типа больше std::max_align_t
. Для образца, foo
выровнен в этом фрагменте кода, поскольку его требование выравнивания составляет 32 байта, но std::max_align_t
имеет выравнивание 16 байт в большинстве реализаций.
struct alignas(32) foo {
char elems[32];
};
|
Оператор new not perloaded для возможно переопределенного класса не повышает дефект, если вы не перегрузите оператор new/new[]
и вы используете версию C++ 17 или более позднюю Стандартную. Оператор по умолчанию new/new[]
в C++ 17 или более поздней версии поддерживает выравнивание путем передачи требования выравнивания в качестве аргумента типа std::align_val_t
, для образца void* operator new(std::size_t size, std::align_val_t alignment)
.
РискОператор по умолчанию new/new[]
выделяет хранилище с требованиями выравнивания std::align_val_t
самое большее. Если вы не перегружаете оператор, когда вы создаете объект с выровненным выше типом, результирующий объект может быть смещен. Доступ к этому объекту может привести к ошибкам доступа или ненормальным завершениям программы.
ЗафиксироватьЕсли вы используете версию C++ 14 или более раннюю версию Standard, передайте требование выравнивания по выровненным типам оператору new/new[]
путем перегрузки оператора.
Пример - Выделенная память меньше, чем требование выравнивания типа foo
#include <new>
#include <cstdlib>
#include <iostream>
struct alignas(64) foo {
char elems[32];
};
foo* func()
{
foo* bar = 0x0;
try {
bar = new foo ;
} catch (...) { return nullptr; }
delete bar;
}
В этом примере структура foo
объявляется с требованием выравнивания 32 байта. Когда вы используете оператор по умолчанию new
для создания bar объектов
, выделенная память для bar
меньше, чем требование выравнивания типа foo
и bar
может быть несовпадающим.
Коррекция - задайте перегрузку оператора new
для обработки требования к выравниванию типа foo
Одна возможная коррекция, если вы используете C11 stdlib.h
или POSIX-C malloc.h
, определяет перегруженный оператор new
который использует aligned_alloc()
или posix_memalign()
или получить хранение с правильной выравниванием.
#include <new>
#include <cstdlib>
#include <iostream>
struct alignas(64) foo {
char elems[32];
static void* operator new (size_t nbytes)
{
if (void* p =
::aligned_alloc(alignof(foo), nbytes)) {
return p;
}
throw std::bad_alloc();
}
static void operator delete(void *p) {
free(p);
}
};
foo* func()
{
foo* bar = 0x0;
try {
bar = new foo ;
} catch (...) { return nullptr; }
delete bar;
}