Effective C++学习笔记 条款04:确定对象被使用前已先被初始化

时间:2023-03-09 05:59:11
Effective C++学习笔记 条款04:确定对象被使用前已先被初始化

一、为内置类型对象进行手工初始化,因为C++不保证初始化它们。

二、对象初始化数据成员是在进入构造函数用户编写代码前完成,要想对数据成员指定初始化值,那就必须使用初始化列表。

 class A
{
public:
A(const string &str)
{
m_str = str; //m_str 的初始化是在进入构造函数中用户自定义编写代码前完成的,所以这里的m_str = str,执行的不是初始化,而是赋值
}
private:
string m_str;
};

  在数据成员的默认初始化,然后再在构造函数中进行赋值,这样效率较低。下面就是用户直接给数据成员指定初始化值的例子:

 class A
{
public:
A(const string &str): m_str(str) //这里调用的是string的构造函数,直接指定初始化值
{
}
private:
string m_str;
};

  这时的效率比之前的方法要高。因为前者是初始化再赋值,后者直接用指定的值初始化要初始化的对象了。

三、初始化列表中,对于数据成员的初始化顺序,是按照数据成员的在class中声明的顺序,而不是按照在初始化列表中出现的先后顺序。

 class B
{
public:
B(const string &str1, const string &str2): m_b(str1), m_a(str2) {}
//这里的初始化顺序是先m_a, 后m_b,因为m_a声明在先
private:
string m_a;
string m_b;
};

四、C++对于“定义于不同编译单元的non-local static对象”的初始化次序无明确定义。

  static对象,指寿命从被构造出来直到程序结束为止,因此排除stack和heap-based对象。其中包括global对象,定义于namespace作用域内的对象,在class内,在函数内以及在file作用域内被声明为static的对象。(注:static对象不只是声明为static的对象)。函数内定义的static对象称为local static对象,其他的static对象就称为non-local static对象。

  第一个文件:

 class FileSystem
{
public:

size_t numDisks() cosnt;
};
extern FileSystem tfs;

  第二个文件中:

 class Directory
{
public:
Directory(params)
{

size_t disks = tfs.numDisks(); //
...
}
};

  当你定义Directory tempDir(params);

  这时由于C++对于定义于不同编译单元的non-local static对象”的初始化次序无明确定义。所以可能在定义 tempDir时,tfs还没有初始化,那么程序就会出现问题

五、对于上述的问题,为免除“跨编译单元之初始化次序”问题,可以使用local static对象替换non-loacl static对象。

 class FileSystem {...};
FileSystem& tfs()
{
static FileSystem fs;
return fs;
}
class Directory
{
Directory(params)
{

size_t disk = tfs().numDisks(); //这样就保证每次使用的时候fs一定初始化了
...
}
};