c++ - reinterpret_cast/static_cast 和未定义的行为

标签 c++ c++11 variant reinterpret-cast static-cast

在一个变体类中,我正在处理原始存储是一个字符数组:

alignas(/* the strictest alignment of all types of the variant */)
char storage[/* ... */];

赋值运算符是这样的:

template<class X>
void operator=(const X &x)
{
  // ...code for clearing the storage and setting the tag for type X...

  new(storage) X(x);
}

而获取存储对象的代码是:

template<class X>
const X &get()
{
  // ...
  return *reinterpret_cast<X *>(storage);
  // ...
}

它似乎有效,但它总是定义明确吗?我担心安全地取消引用指针(类型别名规则是否允许这样做?)。

目前的实现和现在的有什么区别吗

 return *static_cast<const X *>(static_cast<const void *>(storage));

相关问题/答案:

https://stackoverflow.com/a/7321815/3235496 (参见 James Kanze 的评论)。


编辑

第二个问题在这里已经有了答案:C++ When should we prefer to use a two chained static_cast over reinterpret_cast

最佳答案

作为storage正确对齐,我无法想象哪里会出现问题。关于指针转换的段落 (*) 4.10 说:“指向 cv T 的指针”类型的纯右值,其中 T 是对象类型,可以转换为“指向 cv void 的指针”类型的纯右值。将指向对象类型的指针的非空指针值转换为“指向cv void的指针”的结果表示与原始指针值在内存中相同字节的地址

关于你的第二个问题,第5.2.10段 在 reinterpres_cast : 对象指针可以显式转换为不同类型的对象指针。当对象指针类型的纯右值 v 转换为对象指针类型“指向 cv T 的指针”时,结果为 static_cast<cv T*>(static_cast<cv void*>(v)) 其中cv代表可选的 constvolatile .

所以这部分是根据规范保证的。正如我们看到的那样,强制转换为 void *应该指向内存的第一个字节,我对标准的理解没有 UB ...提供编译器有相同的理解;-)

(*) 引用:Draft for current C++ specification

关于c++ - reinterpret_cast/static_cast 和未定义的行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28381338/

相关文章:

c++ - 初始化引用成员:语法不同的原因

c++ - 是否可以通过重复的 std::tuple 的类型名称获取?

c++ - 使用 ADO 将原始文件数据插入 BLOB ("OLE Object") 访问表的字段

c++ - 如何处理 "variant time"(日期, double ,8 字节)?

java - JAXB:无法解码 JAXBElement<?>,取而代之的是获取 ElementNSImpl

c++ - 防止 const 类函数在引用成员上调用非 const 类函数

c++ - 删除前将指针设置为 NULL

c++ - 无法编译使用 libxml++ 的简单程序,因为找不到 glibmmconfig.h

c++ - 是否有检查前提条件的原子增量,即原子值小于指定值?

c++ - Windows 与 Linux 上 std::string 的可移植性和长度