在读条款1的时候真心是没有读懂多少。涉及的是数据抽象!“类型”是一组操作,“抽象数据类型”则是一组具有某种实现的操作。在数据类型设计的时候遵循以下步骤:1、为类型选择一个描述性的名字。这就是对于自己想要实现什么的清楚认识。
2、列出类型所要能执行的操作。定义一个抽象数据类型的依据是能用它做什么。要记住数据的初始化(构造函数),清理(析构函数),复制以及转换(不带explicit关键字修饰的单参数构造函数和转换操作符)。列举类型所能执行的操作。......要避免在实现时简单地为数据成员提供一串get/set操作——那不叫做数据抽象而是懒惰且缺乏想象力的表现。
【explicit用来防止由构造函数定义的隐式转换。】
要明白它的作用,首先要了解隐式转换:可以用单个实参来调用的构造函数定义了从形参类型到该类类型的一个隐式转换。
例如:
class things
{
public:
things(const std::string &name = ""):
m_name(name),height(0),weight(10){}
int CompareTo(const things & other);
std::string m_name;
int height;
int weight;
};
这里things的构造函数可以只用一个实参完成初始化。所以可以进行一个隐式转换,像下面这样:
things a;
................//在这里被初始化并使用。
std::string nm = "book_1";
//由于可以隐式转换,所以可以下面这样使用
int result = a.CompareTo(nm);
这段程序使用一个string类型对象作为实参传给things的CompareTo函数。这个函数本来是需要一个tings对象作为实参。现在编译器使用string nm来构造并初始化一个
things对象,新生成的临时的things对象被传递给CompareTo函数,并在离开这段函数后被析构。
这种行为的正确与否取决于业务需要。假如你只是想测试一下a的重量与10的大小之比,这么做也许是方便的。但是假如在CompareTo函数中还涉及到了要除以初始化为0的height属性,那么这么做可能就是错误的。需要在构造tings之后更改height属性不为0。所以要限制这种隐式类型转换。
那么这时候就可以通过将构造函数声明为explicit,来防止隐式类型转换。
explicit关键字只能用于类内部的构造函数声明上,而不能用在类外部的函数定义上。现在things类像这样:
class things
{
public:
explicit things(const std::string &name = ""):
m_name(name),height(0),weight(0){}
int CompareTo(const things & other);
std::string m_name;
int height;
int weight;
};
这时再进行编译,在vs2008上会提示:没有可用于执行该转换的用户定义的转换运算符,或者无法调用该运算符。
这时你仍然可以通过显示使用构造函数完成上面的类型转换:
things a;
................//在这里被初始化并使用。
std::string nm = "book_1";
//显示使用构造函数
int result = a.CompareTo(things(nm));
google的c++规范中提到explicit的优点是可以避免不合时宜的类型变换,缺点无。所以google约定所有单参数的构造函数都必须是显示的,只有极少数情况下拷贝构造函数可以不声明称explicit。例如作为其他类的透明包装器的类。
effective c++中说:被声明为explicit的构造函数通常比其non-explicit兄弟更受欢迎。因为它们禁止编译器执行非预期(往往也不被期望)的类型转换。除非我有一个好理由允许构造函数被用于隐式类型转换,否则我会把它声明为explicit。
3、为类型设计接口。即为以简单理解为类的接口,抽象数据类型是对语言的扩展。
4、实现类型。“不要让实现影响类型的接口。要实现类型的接口所承诺的约定,大多数情况下,对抽象数据类型的实现的改动,远比对其接口的改动来得频繁”。这似乎是说,接口的设计不要过于紧。ps.这点确实不是很了解。。。