【译】C++ POD的理解(1):aggregates

在阅读《深入理解C++11》时对POD的理解有些疑惑,stack overflow上有一篇高分回答写得非常棒,现在我把它翻译一遍加深一下自己的理解(原文),该译文的其他内容:
【译】C++ POD的理解(2):PODs

如何阅读这篇文章

这篇文章有点长,如果你想同时了解aggregates和PODs(Plain Old Date),就请花点时间把这篇文章读完。如果你仅仅对aggregates感兴趣,则只需阅读第一部分就好。如果你只对PODs感兴趣,你必须首先搞清楚aggregates的定义、含义和示例,然后你就可以跳到PODs部分阅读,但我依然建议还是把第一部分阅读完毕。Aggregates的概念对于定义PODs来说是必不可少的。

什么是aggregates以及它的特殊性

在C++标准中的正式定义(C++03 8.5.1 §1):

Aggregate是一个数组或者是一个没有用户定义的构造函数、没有private和protected的非静态成员变量、没有基类和虚函数的类。

让我们来解析一下这个定义。首先,任何数组都是一个aggregate类型。一个类如果满足以下条件也可以是aggregate类型。。。等等,我们好像忘记说结构体和联合体,它们可以成为aggregate类型吗?答案是他们也可以成为aggregate类型。在C++中,class术语指的是所有的类、结构体和联合体。所以说,一个类(或者结构体,或者联合体)满足上述定义的条件时,就是aggregate类型的。这些条件意味着什么?

  • 这并不意味着aggregate类不能拥有构造函数,事实上它可以拥有默认的构造函数和/或赋值构造函数,只要它们是编译器隐式声明,而不是用户显示声明;
  • 没有private和protected的非静态成员变量,你可以定义很多private和protected的成员函数(构造函数除外)和private和protected的静态成员变量,这都不违背aggregate的规则;
  • aggregate类可以拥有用户声明/用户定义的赋值操作和/或析构函数;
  • 数组都是aggregate类型,即便数组元素是非aggregate类型。
    现在让我们来看一些例子:
class NotAggregate1
{
  virtual void f() {} //remember? no virtual functions
};

class NotAggregate2
{
  int x; //x is private by default and non-static 
};

class NotAggregate3
{
public:
  NotAggregate3(int) {} //oops, user-defined constructor
};

class Aggregate1
{
public:
  NotAggregate1 member1;   //ok, public member
  Aggregate1& operator=(Aggregate1 const & rhs) {/* */} //ok, copy-assignment  
private:
  void f() {} // ok, just a private function
};

你已经明白了aggregates的含义。现在让我们来看下aggregates有什么特别之处。与非aggregate类不同的是,aggregates类型可以用{}进行初始化。这种初始化语法通常被用于数组,但我们现在了解到的是aggregates。所以让我们先从数组开始:

Type array_name[n] = {a1, a2, …, am};

如果 m==n
第i个元素用ai初始化;
如果 m<n
前m个元素分别用a1、a2、a3... am;剩余n-m个元素如果可能的话会用值初始化(后面会解释这个名词);
如果 m>n
编译器将报出一个错误;
其余的情况如a[] = {1,2,3};
会假设数组(n)的大小为m,因此int a[] = {1,2,3}; 等同于a[3] = {1,2,3};

当一个对象是标量类型(bool, int, char, double, 指针等)时,它的值初始化指得是用0进行初始化(bool类型值为false,double类型值为0等)。

数组初始化的例子:

class A
{
public:
  A(int) {} //no default constructor
};
class B
{
public:
  B() {} //default constructor available
};
int main()
{
  A a1[3] = {A(2), A(1), A(14)}; //OK,n == m
  A a2[3] = {A(2)}; //ERROR,A没有默认构造函数,不可以用值初始化a2[1]和a2[2]
  B b1[3] = {B()}; //OK, b1[1] and b1[2]用值初始化,在这个例子中用的默认构造函数
  int Array1[1000] = {0}; //所有的元素都被初始化为0
  int Array2[1000] = {1}; //只有第1个元素初始化为1,其余的元素都初始化为0
  bool Array3[1000] = {}; //大括号可以为空,所有元素初始化为false
  int Array4[1000]; //没有初始化,这和空初始化{}不同
  //在这个例子中元素没有值初始化,拥有不确定的值
  //(除非Array4是一个全局数组)
  int array[2] = {1, 2, 3, 4}; //ERROR,太多元素被初始化
}

现在让我们看下如何使用打括号初始化aggregates类。与上述大致相同。我们将按照非静态数据成员在类定义中的出现顺序初始化它们,而不是数组元素(根据定义,它们都是公共的)。如果需要初始化的比成员变量少,那么其他元素都是进行值初始化。如果无法对未进行显式初始化的某个成员变量进行值初始化,则在编译时会遇到错误。如果初始化成员变量的数量超过所需的数量,我们还是会在编译的时候遇到错误。

struct X
{
  int i1;
  int i2;
};
struct Y
{
  char c;
  X x;
  int i[2];
  float f; 
protected:
  static double d;
private:
  void g(){}      
}; 

Y y = {'a', {10, 20}, {20, 30}};

在上述例子中y.c被初始化为'a'y.x.i110y.x.i220y.i[0]20y.i[1]30y.f则被值初始化,即被初始化为0.0protectedstatic类型的成员变量没有被初始化。

Aggregate联合体则不同,只有第一个成员变量可以用大括号进行初始化。我认为,如果您在c++方面高级到可以考虑使用union(它们的使用可能非常危险,必须仔细考虑)的地步,那么您可以自己在标准中查找union的规则:-)。

现在我们已经知道了aggregates类型的特殊之处,现在我们尝试理解一下它对类型的限制,也就是说为什么会有这些限制。我们应该理解,带大括号的成员初始化意味着类只不过是其成员变量的集合。如果用户定义了构造函数,这意味着徐虎需要在初始化成员变量时需要做额外的工作,所以使用大括号初始化会出错。如果存在虚函数,则意味着该类的对象有一个叫做虚函数表的指针(在大多数实现当中),该指针会在构造函数中进行设置,因此使用大括号初始化是不够的。作为练习,你可以用这种方法理解其他条件限制的含义:-)。

未完待续。。。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容