根据 MSVC,具有易失性成员的结构不再是 POD
Struct with a volatile member no longer a POD according to MSVC
#include <cstdint>
#include <type_traits>
extern "C"
{
struct thread_shared_struct
{
volatile uint32_t ccr;
};
struct thread_shared_struct getStuff();
}
static_assert(std::is_pod<thread_shared_struct>::value, "thread_shared_struct isn't a POD");
int main(int argc, char** argv)
{
thread_shared_struct m = getStuff();
return 0;
}
在 godbolt 上查看此代码
当然,这个例子没有多大意义,但它是从更大的代码库中对我的问题的最小再现。
使用GCC(主干(和Clang(后备箱(,这工作正常。
对于 MSVC (x64, 19.22(,static_assert失败并且 getStuff(( 生成错误,因为它尝试返回与 C 调用约定不兼容的类型。
如果删除"易失性",一切正常。
MSVC 认为thread_shared_struct不是 POD 类型是错误的吗?
外部"C"部分位于需要保持原样的 C 标头中。我如何在C++使用它?
来自Microsoft文档:
当类或结构既是简单布局又是标准布局时,它是 POD(纯旧数据(类型。
它稍后介绍文本类型,包括以下条件:
此外,其所有非静态数据成员和基类必须是文本类型,而不是易失性。
页面上的其他任何地方都没有提到"易失性"。
这一切都符合我们在标准中找到的内容。
因此,我得出结论,这是一个编译器错误。
getStuff(( 生成错误,因为它尝试返回与 C 调用约定不兼容的类型。
实际上,这只是一个警告(C4190(,如果需要,可以禁用它。Visual Studio on x86_64 只有一个调用约定(在此处描述(。您的代码仍然可以正常工作。VS 只是警告您,如果您实际尝试在 C 中使用它,该类型将不起作用。extern "C"
并不意味着编译为 C。
但是,收到此警告确实表明该错误确实存在于编译器中,而不仅仅是在std::is_pod
的实现中。
此外,我建议在新代码中避免使用 POD 术语和std::is_pod
特征,因为它们从 C++20 开始已弃用。
外部"C"部分位于需要保持原样的 C 标头中。我怎样才能从C++使用它?
任何实际上不需要类型符合VS对"POD"的定义的东西,尽管类型特征,都应该没问题。
- 如何循环打印顶点结构
- 通过方法访问结构
- 使用不带参数的函数访问结构元素
- 预处理器:插入结构名称中的前一个行号
- 为什么在没有显式默认构造函数的情况下,将另一个结构封装在联合中作为成员的结构不能编译
- 孤立代码块在结构中引发异常
- 有什么方法可以遍历结构吗
- 如何在 C# 中映射双 C 结构指针?
- 如何在C++中使用结构生成映射
- 无法将结构注册为增强几何体3D点
- 多成员Constexpr结构初始化
- C++将文本文件中的数据读取到结构数组中
- 如何重构类层次结构以避免菱形问题
- 如何在C++中序列化结构数据
- std::vector的包装器,使数组的结构看起来像结构的数组
- 没有为自己的结构调用列表推回方法
- 奇怪的结构&GCC&clang(void*返回类型)
- 在 c++ 中拥有一组结构的正确方法是什么?
- vscode g++链路故障:体系结构x86_64的未定义符号
- 根据 MSVC,具有易失性成员的结构不再是 POD