前言

因为要整理近期学习的c++特性,特地出一篇来介绍POD类型和c++11引进的TrivialStandard-layout

聚合

聚合是以下类型之一:

  • 数组类型
  • 类类型(通常,struct或union)具有
没有用户声明的构造函数 (直到 C++11)
没有用户提供的构造函数(允许显式默认或删除的构造函数) (C++11 起) (C++17 前)
没有用户提供的、继承的或显式(explicit,c++17特意新加)的构造函数(允许显式默认或删除的构造函数) (C++17 起) (C++20 前)
没有用户声明或继承的构造函数(相当于=default不行了) (C++20 起)
  • 没有virtualprivateprotected (C++17 起)基类
  • 没有虚拟成员函数
没有默认的成员初始化器 (C++11 起) (C++14 前)

聚合初始化。它是列表初始化 (C++11 起)或直接初始化 (C++20 起)的一种形式

POD(Plain Old Data)

POD规范着对象的类型,主要是为了兼容C,C++可以直接使用C库函数操作POD数据类型,拥有POD特征的类或结构体通过直接字节拷贝或二进制拷贝后依然能保持数据结构不变,POD类型在C和C++间的操作总是安全的。

特征:

  • 标量类型(算术类型(整型/浮点型)、指针、成员指针、枚举类型)
  • 类类型(class、struct、union)
    • 从c++11起

      • 是平凡(trivial)类型(后续介绍)
      • 是标准布局(standard-layout)类型(后续介绍)
      • 所有非静态成员都是POD类型

特性:

  • 完全与C兼容,但是仍然可以有成员函数;POD类型标准到甚至可以与其他语言兼容;
  • 可以用std::memcpy拷贝(对于非POD类型,即使满足TriviallyCopyable,用std::memcpy拷贝的行为也是未定义的)
  • 有更长的生命周期,从资源获取到资源释放,而非POD类型的是从构造函数结束到析构函数结束;
  • POD类型对象的前部没有填充字节,即对象指针与第一个成员的指针是相等的

来自cppreference:PODType

平凡类型(TrivialType)

要求

来自cppreference:TrivialType

平凡可复制(TrivialCopyable)

要求

以下类型统称为平凡可复制类型

  • 标量类型
  • 可简单复制的类,即满足以下要求的类:
    • 至少一个拷贝构造函数、移动构造函数、复制赋值运算符或移动赋值运算符符合条件
    • 每个符合条件的拷贝构造函数(如果有的话)都是平凡的
    • 每个符合条件的移动构造函数(如果有)都是平凡的
    • 每个符合条件的复制赋值运算符(如果有)都是平凡的
    • 每个符合条件的移动赋值运算符(如果有)都是平凡的
    • 有一个平凡的的未删除析构函数
      • 这里有很多符合条件的要求,但大致如下

        • 如果未删除函数(拷贝构造、移动构造、拷贝赋值、移动赋值 (标准c++20前)),则它是符合条件的。符合条件的函数(上述四者)的平凡性决定了该类是否是隐式生命周期类型,以及该类是否是可平凡复制的类型
      • 这里有很多平凡的概念,但大致如下
        • 它不是用户提供的(意思是,它是被隐式定义或默认的)
        • 所在类没有虚拟,包括虚基类和虚成员函数
        • 对每个非静态类型类型(或类类型数组)成员,递归该要求
  • 平凡可复制对象 的数组

这意味着一个平凡可复制的类没有虚函数虚基类

来自cppreference:TriviallyCopyable

使用is_trivially_copyable(C++11)可判断类型是否是平凡可复制的

std::is_trivially_copyable

对于某些函数的补充说明

平凡拷贝构造函数

平凡可复制对象可以通过手动复制其对象表示来复制,例如使用std::memmove。所有与 C 语言兼容的数据类型(POD 类型)都可以轻松复制。

符合条件的移动构造函数

平凡移动构造函数是执行与普通拷贝构造函数相同的操作的构造函数,也就是说,就像通过std::memmove一样制作对象表示的副本。所有与 C 语言兼容的数据类型(POD 类型)都可以轻松移动。

符合条件的拷贝赋值运算符

平凡拷贝赋值运算符生成对象表示的副本,就像通过std::memmove一样。所有与 C 语言兼容的数据类型(POD 类型)都可以简单地拷贝分配。

符合条件的移动赋值运算符

平凡移动赋值运算符执行与平凡拷贝赋值运算符相同的操作,即生成对象表示的副本,就像std::memmove一样。所有与C语言兼容的数据类型(POD类型)都可以简单地移动赋值。

符合条件的析构函数

平凡析构函数是不执行任何操作的析构函数。具有普通析构函数的对象不需要删除表达式,并且可以通过简单地释放它们的存储来处理。所有与 C 语言兼容的数据类型(POD 类型)都可以轻松破坏。

标准布局类型(Standard-layout Type)

标准布局规范着对对象的布局。标准布局类型对于与用其他编程语言编写的代码进行通信很有用。

当类或结构不包含某些C++语言功能(例如无法在C语言中找到虚函数),并且所有成员都具有相同的访问控制时,该类或结构为标准布局类型。可以在内存中对其进行复制,并且布局已经充分定义,可以由C程序使用。标准布局类具有用户定义的特殊成员函数。此外有以下特征

  • 所有非静态数据成员具有相同的访问控制
  • 没有虚函数或虚基类
  • 没有引用类型的非静态数据成员
  • 类类型的所有非静态成员和基类本身都是标准布局类型
  • 没有与第一个非静态数据成员类型相同的基类
  • 满足以下条件之一:
    • 最底层派生类中没有非静态数据成员,并且具有非静态数据成员的基类不超过一个(换言之继承树中最多只能有一个类有非静态数据成员),或者
    • 没有含非静态数据成员的基类

举两个例子,Base类和Derived类中都有非静态数据成员,因为当Derived继承于Base,有std::is_standard_layout<Derived>为falsestd::is_standard_layout<Base>为true

struct Base
{
int i;
int j;
}; // std::is_standard_layout<Derived> == false!
struct Derived : public Base
{
int x;
int y;
};

Derived 是标准布局,因为 Base 没有非静态数据成员:

struct Base
{
void Foo() {}
}; // std::is_standard_layout<<Derived> == true
struct Derived : public Base
{
int x;
int y;
};

标准布局兼容

涉及两个或两个以上满足标准布局的数据结构兼容问题,概括起来有点复杂,先直接抛cppreference链接看吧(后续再细看),在标准布局内容下边

例子

类A满足POD类型要求,即可直接通过字节拷贝 拷贝其数据,在此情形下,字节拷贝效率是很快的

class A
{
public:
int a;
int b;
}; int main()
{
A a1;
a1.a = 10;
a1.b = 20; char* p = new char[sizeof(A)];
memcpy(p, &a1, sizeof(A)); A* a2 = reinterpret_cast<A*>(p);
cout << a2->a << "\n" << a2->b << "\n";
}

总结

POD概念在C98中被提出,在C++20被启用,取而代之的是在C++11引入的TrivialStandard-layout类型,因本文所介绍内容在《深度探索C++对象模型》中会被重点介绍,待后续阅读完此书籍后,再对本文进行更多补充

引用博客

聚合类型与POD类型

C++中的POD类型

《深度探索C++对象模型》

c++ 聚合/POD/平凡/标准布局 介绍的更多相关文章

  1. WPF Step By Step 完整布局介绍

    WPF Step By Step 完整布局介绍 回顾 上一篇,我们介绍了基本控件及控件的重要属性和用法,我们本篇详细介绍WPF中的几种布局容器及每种布局容器的使用场景,当 然这些都是本人在实际项目中的 ...

  2. html5/css3响应式布局介绍及设计流程

    html5/css3响应式布局介绍 html5/css3响应式布局介绍及设计流程,利用css3的media query媒体查询功能.移动终端一般都是对css3支持比较好的高级浏览器不需要考虑响应式布局 ...

  3. 转:CSS3 Flexbox 布局介绍

    转:CSS3 Flexbox 布局介绍 Flexbox是一个用于页面布局的全新CSS3模块功能.它可以把列表放在同一个方向(从左到右或从上到下排列),并且让这些列表能延伸到占用可用的空间.较为复杂的布 ...

  4. Android 基础:常用布局 介绍 & 使用(附 属性查询)

    Android 基础:常用布局 介绍 & 使用(附 属性查询)   前言 在 Android开发中,绘制UI时常需各种布局 今天,我将全面介绍Android开发中最常用的五大布局 含 Andr ...

  5. AndroidStudio制作登录和注册功能的实现,界面的布局介绍

    前言 大家好,给大家带来AndroidStudio制作登录和注册功能的实现,界面的布局介绍的概述,希望你们喜欢 每日一句: Success is connecting with the world a ...

  6. HTML 页面源代码布局介绍

    http://www.cnblogs.com/polk6/archive/2013/05/10/3071451.html 此介绍以google首页源代码截图为例: 从上到下依次介绍: 1.<!D ...

  7. Android五大布局介绍&属性设置大全

    前言 在进行Android开发中,常常需要用到各种布局来进行UI的绘制,今天我们就来讲下Android开发中最常用的五大布局介绍和相关属性的设置. 目录 Android五大布局介绍&属性设置. ...

  8. Python的标准库介绍与常用的第三方库

    Python的标准库介绍与常用的第三方库 Python的标准库: datetime:为日期和时间的处理提供了简单和复杂的方法. zlib:以下模块直接支持通用的数据打包和压缩格式:zlib,gzip, ...

  9. Qt Quick 布局介绍

    在 Qt Quick 中有两套与布局管理相关的类库,一套叫作 Item Positioner(定位器),一套叫作 Item Layout(布局). 定位器包括 Row(行定位器).Column(列定位 ...

随机推荐

  1. postgreSQL更改表的数据类型

    更改表的sql语句 ALTER table employees ALTER COLUMN status TYPE boolean USING status::boolean; 报下列错误 因为字段上设 ...

  2. 详谈 Java工厂 ---工厂方法模式

    1.前言 有个场景,消费者需要付钱,有可能是使用支付宝.微信.银行卡,那么该怎么选择呢? 是不是想到了使用用if else判断?还是使用switch? 一个地方这样写还好,如果有很多这样的业务,难道都 ...

  3. .gitignore文件编写规则

    1.gitignore说明 在使用git的过程中,一般我们总会有些文件无需纳入git的管理,也不希望它们总出现在未跟踪文件列表,这些文件通常是日志文件.临时文件.编译产生的中间文件.工具自动生成的文件 ...

  4. Scala语言介绍一

    为什么学习scala语言 Scala是基于JVM的语言,与java语言类似,java语言是基于JVM的面向对象的语言,Scala也是基于JVM,同时支持面向对象和面向函数的编程语言.Spark底层的源 ...

  5. 《剑指offer》面试题22. 链表中倒数第k个节点

    问题描述 输入一个链表,输出该链表中倒数第k个节点.为了符合大多数人的习惯,本题从1开始计数,即链表的尾节点是倒数第1个节点.例如,一个链表有6个节点,从头节点开始,它们的值依次是1.2.3.4.5. ...

  6. 《剑指offer》面试题66. 构建乘积数组

    问题描述 给定一个数组 A[0,1,-,n-1],请构建一个数组 B[0,1,-,n-1],其中 B 中的元素 B[i]=A[0]×A[1]×-×A[i-1]×A[i+1]×-×A[n-1].不能使用 ...

  7. vue.config.js报错cannot set property "preserveWhitespace" of undefined

    vue.config.js报错cannot set property "preserveWhitespace" of undefined 最近在项目中配置webpack,由于vue ...

  8. Solon Web 开发,十二、统一的渲染控制

    Solon Web 开发 一.开始 二.开发知识准备 三.打包与运行 四.请求上下文 五.数据访问.事务与缓存应用 六.过滤器.处理.拦截器 七.视图模板与Mvc注解 八.校验.及定制与扩展 九.跨域 ...

  9. 第57篇-profile实例

    之前已经介绍过回边计数和ProfileData与Layout,下面举个具体的例子看下MethodData是怎么利用ProfileData等记录详细的运行时信息的.实例如下: package com.t ...

  10. gin中提供静态文件服务

    package main import ( "github.com/gin-gonic/gin" "net/http" ) func main() { // 静 ...