c++ trivial, standard layout和POD类型解析
1. trivial类型
占用一片连续的内存,编译器可以重排成员变量的顺序或者增加一些padding(为了对齐),因此,可以放心的使用memcpy等函数,
但是,在c代码里面使用可能会出问题(因为可能会被重排),有如下特点:
- 没有虚函数和虚基类
- 基类也必须保证没有non-trivial的构造/析构/operator函数
- 成员变量的类型也必须保证没有non-trivial的构造/析构/operator函数
- 成员变量可以有public/protected/private修饰,且可以出现多个。
- 构造/operator/析构可以有自定义的,但是必须有默认的
可以看出来,trivial主要是针对于 构造/operator/析构 三种函数限制的,普通函数只要不是虚函数,就没有任何限制。
题外话,关于重排序,来看一个例子:
struct Foo {
A a;
B b;
C c;
private:
D d;
E e;
F f;
};
一般来讲,abc的顺序是不会重排的,def的顺序是不会重排的,但是,因为abc和def拥有着不同的访问控制符,是有可能def重排到abc前面的。官方解释
如下:
Nonstatic data members of a (non-union) class declared without an intervening access-specifier are allocated
so that later members have higher addresses within a class object.
The order of allocation of nonstatic data members separated by an access-specifier is unspecified (11.1)
注意,静态成员变量不影响
好了,定义部分讲完了,来看几个例子:
//trivial类型,同时也是pod类型(后面会解释)
class TrivialClass{
int a_;
bool flag_;
//bool flag2_{false}; //这种就不是trivial了
};
//不是trivial类型
class NonTrivialClass{
int a_;
bool flag_;
bool flag2_{false}; //比上面就多了一个初始化赋值,就不是trivial了,但是这个是stdlayout(后续会介绍)
};
class NonTrivialClass2{
int a_;
bool flag_;
bool flag2_ = false; //同NonTrivialClass
};
//不是trivial类型
class NonTrivialClass3{
NonTrivialClass3() {} //和NonTrivialClass3() = default的写法是不一样的,决定是否为trivial类型
int a_;
bool flag_;
bool flag2_ = false; //同NonTrivialClass
};
//下面的也满足trivial要求
class TrivialClass2{
public:
//注意,静态类型是不影响判断的,因为静态成员变量不会存储在结构体/类里面
static std::string static_member;
//也可以有构造函数,但是必须有默认的构造函数才行
TrivialClass2(int a, int b): a_(a), b_(b) {}
TrivialClass2() = default;
void foo() {}
int a_;
private:
int b_;
bool flag_;
};
虽然举的例子都是没有继承的简单类,但是如果有继承,也有可能是trivial类型,需要保证基类也是trivial类型
2. standard layout类型
standard layout类型(以下简称stdlayout):不包含任何c语言里没有的功能,可以使用memcpy等函数,并且可以在c代码里放心使用,这点不同于trvial的类型,同时,stdlayout类型也能拥有自己的函数,有如下特点:
- 没有虚函数和虚基类
- 所有的非静态成员变量必须拥有完全一样的public/protected/private访问控制符, 比如下面的
TrivialClass2
就不是stdlayout类型 - 所有非静态成员变量也必须是stdlayout类型
- 所有的基类也必须是stdlayout
- 不能把基类类型作为第一个非静态成员变量
- 下面的条件满足一个即可(总结就是要么子类有非静态成员变量,要么父类有):
- 子类没有非静态成员变量并且只有一个基类有非静态成员变量
- 基类都没有非静态成员变量
注意,静态成员变量不影响
可以看出来,stdlayout放宽了构造/operator/析构的限制,但是也收紧了访问控制符的限制(只能有一种)
接下来看代码示例:
//满足stdlayout的要求,但是因为没有默认的构造函数,因此不满足trivial的要求
class StdLayoutClass{
public:
StdLayoutClass(int a, int b): a_(a), b_(b) {}
int a_;
int b_;
};
//和上面的结构体就多了一个private(权限修饰符不一样),这里就不是stdlayout了
class ClassMixedAccess{
public:
ClassMixedAccess(int a, int b): a_(a), b_(b) {}
int a_;
private:
int b_;
};
class StdLayoutBase{
public:
StdLayoutBase() {} //和StdLayoutBase()=default的写法是有区别的,决定了是否为trivial
int a_;
int b_;
};
//这个不是standard layout,因为基类有非静态成员变量,也不是trivial,因为积累的构造函数不是default(注意,和=default的写法是有区别的)
class StdLayoutDerived : public StdLayoutBase {
int c_;
int d_;
};
class StdLayoutBase2{
StdLayoutBase2() = default;
};
//这个就满足stdlayout的要求了,因为必须满足子类或者基类只能有一个有非静态成员变量
class StdLayoutDerived2 : public StdLayoutBase2 {
StdLayoutDerived2() = default;
int c_;
int d_;
//可以拥有自己的函数
void test_func(){
printf("hello world");
}
};
3. 集大成者,POD(Plain Old Data)类型
POD类型:当一个类既是trivial又是stdlayout类型的时候,这个时候它就是pod类型,因此,pod类型的特点如下:
- 拥有着连续的内存
- 每一个成员变量的地址都高于其前一个成员变量(也就是没有进行重排序操作)
- 同样的,也有嵌套要求,非静态成员变量必须也是POD类型
综上,POD类型可以在I/O操作中放心的进行copy和恢复, 基本类型比如int,char,引用等都是POD类型。
注意,静态成员变量不影响。
示例代码如下:
class ClassWithVirtual{
public:
virtual bool foo() {}
};
//出现了虚函数,因此,trivial和stdlayout都不是
class ClassWithVirtualDerived : public ClassWithVirtual {
public:
int a_;
int b_;
virtual bool foo() override{}
};
//有不同的访问限定符,因此,属于trivial但是不属于stdlayout
class ClassDiffAccess{
public:
int a_;
private:
int b_;
};
//拥有着自定义的构造函数(但是没有显示指定default),因此不属于trivial
class ClassWithoutDefaultConstructor{
public:
ClassWithoutDefaultConstructor() {};
int a_;
int b_;
};
//集大成者,既属于trivial也属于stdlayout,也就是POD类型
class PODClass {
int a_;
int b_;
};
4. 测试代码
上述类,可以通过如下代码测试
#define BOOL_2_STRING(val) ((val) ? "true" : "false")
/**
* tool function
* @tparam T
*/
template<class T>
void test_trivial_stdlayout_pod_type_internal(const char *type_name) {
bool is_trivial = std::is_trivial<T>::value;
bool is_std_layout = std::is_standard_layout<T>::value;
bool is_pod = std::is_pod<T>::value;
printf("%s: is_trivial[%s] is_standard_layout[%s] is_pod[%s]\n",
type_name, BOOL_2_STRING(is_trivial),
BOOL_2_STRING(is_std_layout), BOOL_2_STRING(is_pod));
}
/**
* 测试三大类型,trivial,standard layout, pod
*/
void test_trivial_stdlayout_pod_type() {
test_trivial_stdlayout_pod_type_internal<TrivialClass>("TrivialClass");
test_trivial_stdlayout_pod_type_internal<NonTrivialClass>("NonTrivialClass");
test_trivial_stdlayout_pod_type_internal<NonTrivialClass2>("NonTrivialClass2");
test_trivial_stdlayout_pod_type_internal<NonTrivialClass3>("NonTrivialClass3");
test_trivial_stdlayout_pod_type_internal<TrivialClass2>("TrivialClass2");
test_trivial_stdlayout_pod_type_internal<StdLayoutClass>("StdLayoutClass");
test_trivial_stdlayout_pod_type_internal<ClassMixedAccess>("ClassMixedAccess");
test_trivial_stdlayout_pod_type_internal<StdLayoutDerived>("StdLayoutDerived");
test_trivial_stdlayout_pod_type_internal<StdLayoutDerived2>("StdLayoutDerived2");
test_trivial_stdlayout_pod_type_internal<ClassWithVirtualDerived>("ClassWithVirtualDerived");
test_trivial_stdlayout_pod_type_internal<ClassDiffAccess>("ClassDiffAccess");
test_trivial_stdlayout_pod_type_internal<ClassWithoutDefaultConstructor>("ClassWithoutDefaultConstructor");
test_trivial_stdlayout_pod_type_internal<PODClass>("PODClass");
}
/*
输出如下:
TrivialClass: is_trivial[true] is_standard_layout[true] is_pod[true]
NonTrivialClass: is_trivial[false] is_standard_layout[true] is_pod[false]
NonTrivialClass2: is_trivial[false] is_standard_layout[true] is_pod[false]
NonTrivialClass3: is_trivial[false] is_standard_layout[true] is_pod[false]
TrivialClass2: is_trivial[true] is_standard_layout[false] is_pod[false]
StdLayoutClass: is_trivial[false] is_standard_layout[true] is_pod[false]
ClassMixedAccess: is_trivial[false] is_standard_layout[false] is_pod[false]
StdLayoutDerived: is_trivial[false] is_standard_layout[false] is_pod[false]
StdLayoutDerived2: is_trivial[true] is_standard_layout[true] is_pod[true]
ClassWithVirtualDerived: is_trivial[false] is_standard_layout[false] is_pod[false]
ClassDiffAccess: is_trivial[true] is_standard_layout[false] is_pod[false]
ClassWithoutDefaultConstructor: is_trivial[false] is_standard_layout[true] is_pod[false]
PODClass: is_trivial[true] is_standard_layout[true] is_pod[true]
*/
c++ trivial, standard layout和POD类型解析的更多相关文章
- C++中的Trivial 、POD、non-POD和Standard Layout概念
POD types non-POD types Standard Layout types A Formal Definition Informally, a standard layout clas ...
- C++ trivial和non-trivial构造函数及POD类型(转)
原博客地址http://blog.csdn.net/a627088424/article/details/48595525 最近正纠结这个问题就转过来了,做了点补充(参考<深度探索C++对象模型 ...
- 关于C++ 中POD类型的解析
转自: http://liuqifly.spaces.live.com/blog/cns!216ae3a149106df9!221.entry (C++-98:1.8;5)给出的定义:将对象的各字节拷 ...
- C++ POD 类型
POD 是 C++ 中一个比较重要的概念,POD 是英文 Plain Old Data 的缩写(通俗讲就是类或结构体通过二进制拷贝后还能保持其数据不变),用来描述一个类型(包括 class.union ...
- C++11 POD类型
POD,全称plain old data,plain代表它是一个普通类型,old代表它可以与c兼容,可以使用比如memcpy()这类c中最原始函数进行操作.C++11中把POD分为了两个基本概念的集合 ...
- 3. C++ POD类型
POD全称Plain Old Data,通常用于说明1个类型的属性.通俗的讲,一个类或结构体通过二进制拷贝后还能保持其数据不变,那么它就是一个POD类型. C++11将POD划分为2个基本概念的合集, ...
- 聚合类型与POD类型
Lippman在<深度探索C++对象模型>的前言中写道: I have heard a number of people over the years voice opinions sim ...
- c++11 pod类型(了解)
啥是POD类型? POD全称Plain Old Data.通俗的讲,一个类或结构体通过二进制拷贝后还能保持其数据不变,那么它就是一个POD类型. 平凡的定义 .有平凡的构造函数 .有平凡的拷贝构造函数 ...
- SpringMVC中对多部件类型解析---文件(图片)上传
加入上传图片jar包 commons-io-2.4.jar commons-fileupload-1.3.jar 在页面form中提交enctype="multipart/form-data ...
- 关于POD和非POD类型中,list initialization和constructor initialization(未解决)
如果你的成员是POD类型的,那么list initialization和constructor initialization没有任何区别 #include<iostream> using ...
随机推荐
- 使用 Loki 收集 Traefik 日志
转载自:https://mp.weixin.qq.com/s?__biz=MzU4MjQ0MTU4Ng==&mid=2247492264&idx=1&sn=f443c92664 ...
- Deployment控制器(pod)更新策略
最小就绪时间: 配置时,用户可以使用Deplpoyment控制器的spec.minReadySeconds属性来控制应用升级的速度.新旧更替过程中,新创建的Pod对象一旦成功响应就绪探测即被视作可用, ...
- 安装 Ubuntu 教程
1.选择中文安装 2.****到了如下界面,我们点击继续: 3.然后点击,现在安装: 4.****到了这界面,点击继续: 5.如下,输入你的位置,随便输入就好: 6.****然后选择汉语,点击继续: ...
- 企业MES系统与ERP信息集成要素有哪些?
关于要讲明企业MES系统与ERP信息集成要素有哪些,得先弄清楚他们之间的关系:从工厂的管理来说,ERP在上MES在下,ERP统领企业全局包括MES,为管理层服务,重心在于企业决策,ERP对企业宏观管理 ...
- 聊聊SQL注入
SQL注入问题 概述: 首先SQL注入是一个非常危险的操作,很可能被一些不怀好意的人钻空导致我们系统出现异常等状况,比如数据库遭到破坏或被入侵. 原因:使用JDBC的Statement语句添加SQL语 ...
- Go微服务实战 - 从0到1搭建一个类Instagram应用(持续更新)
概要 近几年各大移动应用基本都有社区Community(或动态Moments)的功能,展现形式各不相同,比如 国内的有:微博.朋友圈.抖音.小红书.keep.绿洲.即刻等 国外的有:Instagram ...
- P2216 [HAOI2007]理想的正方形 方法记录
[HAOI2007]理想的正方形 题目描述 有一个 \(a \times b\) 的整数组成的矩阵,现请你从中找出一个 \(n \times n\) 的正方形区域,使得该区域所有数中的最大值和最小值的 ...
- uni-app 如何优雅的使用权限认证并对本地文件上下起手
这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 1.起因 最近有一个需求,需要使用自定义插件,来对接硬件功能,需要配合对手机的权限进行判断和提示,并在对接后对本地文件进行操作,这里给大家 ...
- python-D3-语法入门1
Python语法注释 什么是注释 注释其实就是对一段代码的解释说明(注释是代码之母) 如何编写注释 方式1:解释说明文字前加警号 (pycharm中有快捷键ctrl+?) # 注释(单行注释) 方式2 ...
- JavaScript基本语法(数组与JSON)
5.数组 #①使用new关键字创建数组 // 1.创建数组对象 var arr01 = new Array(); // 2.压入数据 arr01.push("apple"); ar ...