当前位置: 首页 > 编程笔记 >

关于C++类的成员初始化列表的相关问题

张成济
2023-03-14
本文向大家介绍关于C++类的成员初始化列表的相关问题,包括了关于C++类的成员初始化列表的相关问题的使用技巧和注意事项,需要的朋友参考一下

在以下四中情况下,要想让程序顺利编译,必须使用成员初始化列表(member initialization list):

1,初始化一个引用成员(reference member);

2,初始化一个常量对象(const member);

3,调用一个基类的构造函数,且该基类的构造函数有一组参数;

4,调用一个成员类(member class)的构造函数,且该构造函数有一组参数

这四种情况程序可以正常编译,但是效率有所欠缺(下面会具体说到)。

class Word{

String _name;

int _cnt;

public:

Word() {

_name = 0;

_cnt = 0;

}

};

上面这个程序的实现机制是:Word类的构造函数会先生成一个String类的临时对象(注意,_name是String类的对象),然后对该临时对象初始化。

然后通过赋值运算符将临时对象赋给_name,最后析构该临时对象。

以下是构造函数的内部扩张结果,c++伪代码:

Word::Word()

{

_name.String::String();  //调用String类的默认构造函数(default constructor)

String temp = String(0); //产生类的临时对象 并初始化

_name.String::operator = (temp); //通过赋值运算符将临时对象的值(深)拷贝给 _name

temp.String::~String();   //调用String的析构函数

_cnt = 0;

}

以上的代码效率并不高,因为中间需要调用默认构造函数和析构函数生成和销毁一个临时对象,以下是一个更有效率的实现方法:

Word::Word : _name (0)  //_name直接调用String类的构造函数对其赋值

{

_cnt = 0;

}

它会被构造函数扩张成以下的形式(c++++伪代码)

Word::Word()

{

_name.Sting::String(0);  //调用String (int) 构造函数

_cnt = 0;

}

成员初始化列表并不是一组函数调用,编译器一一操作初始化列表,以适当的顺序在构造函数中插入初始化的操作,并且是在程序员显式的写入代码之前进行。

列表的中的项目次序是由类中的成员声明次序决定的,不是由初始化列表中的排列顺序决定。“初始化次序”和“初始化列表中的项目排列顺序”的错乱会带来意想不到的错误:

class X {

  int i;

  int j;

public:

  X (int value) : j (value), i (j)

 {}....

}; 

以上代码编写者的本意是要把j的初值设置为 value, 再把 i 的初值设置为 j 。然而,由于声明次序 i 在 j 之前,初始化列表中 i(j) 实际上比 j(value)更早执行,

这就带来了意想不到的错误。正确的写法应该是:

 class X {

  int i;

  int j;

public:

  X (int value) : j (value)    // j (value) 此处调用构造函数赋初值 

{ i = j; }

};

虽然这种写法仍然是 i 声明在 j 之前,但是并不会发生错误,因为初始化列表中的项目被插入到构造函数中不会再保持原来的声明次序,也就是说初始化列表被插入到构造函数中初始化列表中的项目顺序优先级高于代码编写者显式声明的顺序。

以上这篇关于C++类的成员初始化列表的相关问题就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持小牛知识库。

 类似资料:
  • 我试图理解为什么以下Java代码的输出是这样的: 输出: 我可以想象,当E05SimpleInheritance公共类的主要方法被调用时,应该会发生以下事情 加载非公共类C并初始化其字段(在调用类C的默认构造函数之前) 所以最终的输出应该是B()A(),这显然是错误的,所以我不太明白在这种情况下代码是如何流动的。你能告诉我为什么打印的是A()B()而不是B()A()吗

  • 在C++98标准里,只有static const声明的整型成员能在类内部初始化,并且初始化值必须是常量表达式。这些限制确保了初始化操作可以在编译时期进行。例如: int var = 7; class X { static const int m1 = 7; // 正确 const int m2 = 7; // 错误:无static static int m3 =

  • 问题内容: 我最近刚与Python中的一个错误作斗争。那是那些愚蠢的新手错误之一,但是它让我思考了Python的机制(我是C ++的老程序员,是Python的新手)。我将列出错误的代码并解释如何解决该问题,然后我有两个问题。 场景:我有一个叫做A的类,它有一个字典数据成员,下面是其代码(当然这是简化的): 使用此代码的类为B类: 请注意,每次调用都会初始化类A的新“干净”实例,并在添加前后打印字典

  • 本文向大家介绍关于Java数组声明、创建、初始化的相关介绍,包括了关于Java数组声明、创建、初始化的相关介绍的使用技巧和注意事项,需要的朋友参考一下 本文讲述了Java数组的几个相关的方面,讲述了对Java数组的声明、创建和初始化,并给出其对应的代码。 一维数组的声明方式:type var[]; 或type[] var; 声明数组时不能指定其长度(数组中元素的个数), Java中使用关键字new

  • 初始化类成员的确切时间? 在下面的代码中: B&C类的对象是什么时候创建的?并且它是否保证被创建? 一个专业的app到底该不该用这样的代码?

  • 问题 你想在类被定义的时候就初始化一部分类的成员,而不是要等到实例被创建后。 解决方案 在类定义时就执行初始化或设置操作是元类的一个典型应用场景。本质上讲,一个元类会在定义时被触发, 这时候你可以执行一些额外的操作。 下面是一个例子,利用这个思路来创建类似于 collections 模块中的命名元组的类: import operator class StructTupleMeta(type):