。
#@author: gr#@date: 2014-09-30#@email: forgerui@gmail.com
记录读书过程中一些知识点。可能不系统,
:-)
。
Part 1: C++类(第7章)
一、某些类不能依赖于默认构造函数
默认构造函数:不接受任何实参的构造函数
- 只有当类没有声明任何构造函数时,编译器才会自动地生成默认构造函数。
- 合成的默认操作可能执行错误的操作。 比如:定义在块中的内置类型或复合类型(比如数组和指针)的对象被默认初始化,则它们的值将是未定义的,应该需要手动进行定义。
- 有时候,编译器不能为某些类合成默认的构造函数。 比如:有些成员类没有提供默认构造函数。
二、默认构造函数(C++11)
struct Sales_data{ Sales_data() = default;};
=default
:C++11
标准规定,如果需要默认的行为,那么可以加上= default
来要求编译器生成构造函数。
三、类内成员初始化
struct Sales_data{ //有些编译器不支持 double a = 1.0;};
有些编译器支持内置类型的数据成员提供了初始值,有些编译器不支持类内初始值,那只能使用构造函数初始值列表或构造函数进行初始化。
四、构造函数的初始值有时必不可少
有些类型必须使用初始值进行初始,如引用,const,和没有默认构造函数的类。这些类型无法在构造函数体中初始,只能放到初始值列表中。
此外,初始化和赋值的区别事关底层效率问题,前者直接初始化数据成员,后者先初始化后赋值,效率低。
class ConstRef{ public: ConstRef(int ii); private: int i; const int ci; int &ri;};ConstRef::ConstRef(int ii){ //正确 i = ii; //错误:const类型无法进行赋值 ci = ii; //错误:引用类型无法进行初始化 ri = i;}
五、委托构造函数(C++11)
委托构造函数把工作委托给另一个构造函数,受委托的构造函数的初始值列表和函数体被依次执行。如果,函数体内有代码,会先执行受委托函数的函数体,之后才执行委托者的函数体。
struct Sales_data{ public: //非委托构造函数 Sales_data(std::string s, unsigned cnt, double price):bookNo(s), units_sold(cnt), revenue(cnt*price){} //其余构造函数全都委托给另一个构造函数 Sales_data():Sales_data("", 0, 0){} Sales_data(std::string s):Sales_data(s, 0, 0){} Sales_data(std::istream &is):Sales_data(){ read(is, *this); }};
六、使用默认构造函数
//错误:声明了一个函数而非对象Sales_data obj();//正确:obj2是一个对象而非函数Sales_data obj2;
七、explicit抑制隐式转换
`explicit`只能在类内使用,可以阻止类型进行隐式转换。class Sales_data{ public: //只允许在类内构造函数 explicit Sales_data(std::string s):bookNo(s){}};
八、static成员
static
成员与类关联在一起,不与对象绑定在一起。static
只出现在类内部的声明语句中。