Effective C++读书笔记(四)确认对象在使用之前被初始化

时间:2021-07-15 19:48:34

     前言c++对于变量声明具有很强的变化性,变量初始化的值因编译器而异。比如,int x;在某些情况下x被初始化为0,而有的时候是-858993460(min_int)。对于某些编译器会检查没有被初始化的变量(vc6.0,vs2008)并给出警告,但是做为合格coder应该消除警告。


构造函数初始化效率高于赋值

#include<iostream>
#include<list>
class PhoneNumber{};

class ABEntry{
public:
ABEntry(const std::string& name,const std::string& address,const std::list<PhoneNumber>& phones);
private:
std::string theName;
std::string theAddress;
std::list<PhoneNumber> thePhones;
int numTimesConsulted;
};
ABEntry::ABEntry(const std::string& name,const std::string& address,const std::list<PhoneNumber>& phones){
theName=name;
theAddress=address;
thePhones=phones;
numTimesConsulted=0;}
int main()
{
//ABEntry abc=new ABEntry;
//std::cout<<sizeof(abc)<<std::endl;
return 0;
}

上面代码是用赋值给类成员初始化,声明ABEntry abc=new ABEntry时发生错误。

ABEntry::ABEntry(const std::string& name,const std::string& address,const std::list<PhoneNumber>& phones)
:theName(name),
theAddress(address),
thePhones(phones),
numTimesConsulted(0)
{}

把ABEntry的构造改为上述代码,使用成员初始化列表初始化。需要注意的是成员初始化列表要和类成员声明顺序一致,具体参考http://blog.csdn.net/zxy2011qp/article/details/12842745

构造次序

//head.h
#include<stddef.h>
#include<iostream>
class FileSystem {
public:
std::size_t numDisk() const;
};

//C.cpp
#include "head.h"
FileSystem tfs;

//B.cpp
#include"head.h"
extern FileSystem tfs;

class Directory{
public:
Directory(int params);
};

Directory::Directory(int params)
{
std::size_t disks=tfs.numDisk();
}

//A.cpp
#include<iostream>
#include"head.h"
using namespace std;


int main()
{
Directory s1(1);
return 0;
}

上面四个文件编写过程按上面顺序,最后编译A.cpp出现错误(其他都编译通过),主要是类Directory没有声明,这是因为类Directory依赖于外部变量tfs,而tfs和类Directory的声明顺序是不可知道的(谁先谁后不能确定),所以发生了错误。

我们需要改变的是把tfs声明为静态变量就能解决这个问题,如下

//head.h
#include<stddef.h>
#include<iostream>
class FileSystem {
public:
std::size_t numDisk() const{return 0;}
};
class Directory{
public:
Directory(int params);
};

//B.cpp
#include"head.h"
FileSystem& tfs(){
static FileSystem fs;
return fs;
}



Directory::Directory(int params)
{
std::size_t disks=tfs().numDisk();
}

//A.cpp
#include<iostream>
#include"head.h"
using namespace std;


int main()
{
Directory s1(0);
return 0;
}