Effective C++笔记(3)—条款4分析
2016-08-14 22:23
260 查看
1.条款04:确定对象被使用前已先被初始化
这是一个比较好理解的条款,从刚学习C语言开始,这样的问题就一直伴随至今。1.1 内置类型的初始化
对于int、
double这样的内置类型,需要手动初始化,比如说:
int x= 0; const char* text="A C-style string"; double d; cin >> d;//以输入流的方式初始化
未初始化变量的错误,已经很少犯了,毕竟已经身经百战。
1.2构造函数初始化
这里纠正了我以前的误解,那就是但凡是在构造函数中做的事情就是初始化,然而,书中指出,带参数的构造函数使用成员初值列表的做法可谓之初始化,初始化的顺序则是以成员初值列中的次序为标准(即使声明顺序不一样),而在函数体中的赋值操作,只能算作赋值:
class A { private: int a; public: A(int value):a(value){}//这是初始化 }; //*************** class A { private: int a; public: A(int value){a=value;}//这是赋值 };
其实两者效果没差,不过初始化的结果通常是效率更高。
对于默认构造函数来说,依然是使用成员初值列表的方式,让需要初始化的成员去调用各自的默认构造函数,例如:
class B { private: int b; public: B():b(0){} }; class A { private: string t1; B b; int x; public: //各自调用各自的默认构造函数 A() :t1(), b(), x(0){} };
1.3non-local static对象
什么是non-local static对象
static关键字的作用,在cppreference.com写道:(参见:C++ KeyWords:static)
1.static storage duration with internal linkage specifier(静态存储周期声明)
2.declarations of class members not bound to specific instances(对象无关的成员)
函数内部的static对象被称作local static对象,反之,其他的static对象则是non-local static对象了。要成为static对象的方法有很多,static关键字是其特性的说明之一,正如上述1.所说。
书中描述的static对象,并不是我们常说的什么静态局部变量,静态全局变量什么的,前者指的是其存储周期,其寿命从被构造出来到程序结束为止,后者与变量的存储位置,作用范围有关。比如书中的例子:
extern FileSystem tfs;
是一个非静态全局变量,但是是一个non-local static 对象。
编译单元和示例分析
简单来说,cpp文件和其包含头文件就是一个编译单元。现在的问题是,如果两个不同的编译单元分别包含两个non-local static对象,并且其中一个对象的初始化依赖于另外一个,则这两个对象的初始化顺序是未知的。
举个例子:
//first.cpp #include <iostream> #include "common.h" test1 t1;
//second.cpp #include <iostream> #include "common.h" test2 t2;
//last.cpp #include <iostream> #include "common.h" using namespace std; int main() { cout << "Hello World" << endl; return 0; }
//common.h #ifndef _LIB_H_ #define _LIB_H_ #include <iostream> using namespace std; class test2; extern test2 t2; class test2 { public: test2() { value = 100; cout << "test2 construct" << endl; } void said() { cout << "I am test2 " << "value=" << value << endl; } private: int value; }; class test1 { public: test1() { cout << "test1 construct" << endl; t2.said(); } }; #endif
示例中可以看到,test1实例的构造的前提是t2已经构造。first.cpp和second.cpp则是分别对于t1和t2的构造。t1和t2分别是不同编译单元中的non-local static对象。
#g++ -c -o first.o first.cpp #g++ -c -o second.o second.cpp #g++ -c -o last.o last.cpp #g++ last.o first.o second.o #./a.out test1 construct I am test2 value=0 test2 construct Hello World #g++ last.o second.o first.o #./a.out test2 construct test1 construct I am test2 value=100 Hello World
如果t2后于t1初始化,那么结果将是未定义的。
一个简单设计将可以消除这个问题:将non-local static对象放到其专属的函数内,该对象在函数内部也被声明为static,这些函数返回该对象的引用,这就是C++ 单例模式的一种常见手法。
因此,对上述示例进行修改,将non-local static对象放到函数内部,并返回一个对象引用:
//second.cpp test2& getInstance() { static test2 t2; return t2; }
此时,我们需要t2对象的时候,使用其专属的函数,而不是原来的non-local static对象:
class test1 { public: test1() { cout << "test1 construct" << endl; getInstance().said(); // t2.said();//取代了直接操作对象 } };
这样就提供了初始化顺序的保证,因为对于local static对象来说,这只在函数调用的时候才会初始化,并且当没有初始t1的时候,t2也不会初始化,因而不会引发构造和析构的成本。
2.参考
1.Effective C++2.http://blog.csdn.net/zhangyifei216/article/details/50549703
3.http://www.cnblogs.com/burandanxin/archive/2009/10/16/1584735.html
3.后记
EffectiveC++这本书没有多厚,但看了两天下来发现里面的知识点还是很多,原计划的是一篇博客弄懂2-3个条款,看样子现在不得不视情况而定了,呵呵,加油吧~相关文章推荐
- Effective C++笔记(4)—条款5/6分析
- Effective c++ 学习笔记——之条款1:视C++为一个语言联邦
- Effective c++学习笔记——条款6,不想自动生成函数,要明确拒绝
- effective C++笔记之条款23:必须返回一个对象时不要试图返回一个引用
- effective C++笔记之条款29: 避免返回内部数据的句柄
- effective C++笔记之条款28: 划分全局名字空间
- more effective c++学习笔记 ---- 条款31
- effective C++笔记之条款34: 将文件间的编译依赖性降至最低
- effective C++笔记之条款19:分清成员函数,非成员函数和友元函数
- effective C++笔记之条款18:争取使类的接口完整并且最小
- effective C++笔记之条款31、32: 千万不要返回局部对象的引用,也不要返回函数内部用new初始化的指针所指对象的引用、尽可能地推迟变量的定义
- effective C++笔记之条款5:对应的new和delete要采用相同的形式
- effective C++笔记之条款11: 为需要动态分配内存的类声明一个拷贝构造函数和一个赋值操作符
- effective C++笔记之条款9、10:避免隐藏标准形式的new、如果写了operator new 就要同时写operator delete
- effective C++笔记之条款20、21:避免public接口出现数据成员、尽可能使用const
- Effective c++ 学习笔记——条款04:确定对象被使用前已先被初始化
- effective C++笔记之条款8: 写operator new和 operator delete 时要遵循常规
- effective C++笔记之条款13: 初始化列表中成员列出的顺序和它们在类中声明的顺序相同
- effective C++笔记之条款26:当心潜在的二义性
- effective C++笔记之条款33: 明智地使用内联