C++ 对齐数组或 64 字节边界上的“unsigned long”向量,以及相关的 delete[]

问题描述

我试过了

unsigned long* ulongsArray = new(std::align_val_t{ 64 }) unsigned long[1024];
delete[] ulongsArray;
// ::operator delete[](ulongsArray,std::align_val_t{ 64 });   // same compiler error
// ::operator delete[](new(std::align_val_t{ 64 }) unsigned long[1024],std::align_val_t{ 64 });   // same compiler error

但是,在 VisualStudio 2019 下收到错误错误 C2956:通常的解除分配函数‘void operator delete[](void *,std::align_val_t) noexcept’将被选为放置解除分配函数。”这似乎指向 C++17 标准缺少对齐数组的 delete[] 定义。

VisualStudio2019 似乎在认情况下与英特尔编译器一起在 64 字节边界上对齐。但是,g++ 似乎对齐到 16 字节边界,我需要 64 字节对齐(缓存行)。

解决方法

这似乎是 MSVC 编译器的一个问题,因为它可以用 gcc 和 clang 很好地编译。过去曾报道过here

但是,通过执行以下操作,可以实现您正在寻找的预期结果(跨 MSVC、clang 和 GCC):

unsigned long *ulongsArray = static_cast<unsigned long*>(
        operator new[](sizeof(unsigned long) * 64,(std::align_val_t)(64)));
delete[] ulongsArray;