《Effective C++》第三版-4. 设计与声明(Design and Declarations)

条款17:让接口容易被正确使用,不易被误用(Make interfaces easy to use correctly and hard to use incorrectly)

限制类型和值

class Date {
public:
	Date(int month, int day, int year);  //可能月日年顺序错,可能传递无效的月份或日期
	...
};

可使用类型系统(type system)规避以上错误,即引入外覆类型(wrapper type)区别年月日:

struct Day {
explicite Day(int d)
	: val(d) { }
int val;
}
struct Month {
explicite Month(int m)
	: val(m) { }
int val;
}
struct Year{
explicite Year(int y)
	: val(y) { }
int val;
}

class Date {
public:
	Date(const Month& m, const Day& d, const Year& y);  //可能月日年顺序错,可能传递无效的月份或日期
	...
};
Date d(Month(3), Day(30), Year(1995));  //可有效防止接口误用

保证了类型正确之后,需要保证输入的值有效:

class Month {
public:
	static Month Jan() { return Month(1); }
	static Month Feb() { return Month(2); }
	...
	static Month Dec() { return Month(12); }
	...
private:
	explicit Month(int m);
	...
};
Date d(Month::Mar(), Day(30), Year(1995));

规定能做和不能做的事

if ( a * b = c) ...  //以const修饰操作符*,使其不能被赋值

提供行为一致的接口

为了避免忘记删除或者重复删除指针,可令工厂函数直接返回智能指针:

Investment* createInvestment(); //用户可能忘记删除或者重复删除指针
std::tr1::shared_ptr<Investment> createInvestment();

若期望用自定义的getRidOfInvestment,则需要避免误用delete,可考虑将getRidOfInvestment绑定为删除器(deleter):

删除器在引用次数为0时调用,故可创建一个null shared_ptr

std::tr1::shared_ptr<Investment> createInvestment()
{
	std::tr1::shared_ptr<Investment> retVal(static_cast<Investment*>(0), 
																					getRidOfInvestment);  //创建一个null shared_ptr
	retVal = ... ;  //令retVal指向目标对象
	return retVal;
}

若pInv管理的原始指针能在pInv创立之前确定下来,则将原始指针直接传递给pInv的构造函数更好

tr1::shared_ptr会自动使用每个指针专属的删除器,从而无须担心cross-DLL problem:

cross-DLL problem:对象在动态连接程序库(DLL)中被new创建,但在另一个DLL内被delete销毁

//返回的tr1::shared_ptr可能被传递给任何其他DLL
//其会追踪记录从而在引用次数为0时调用那个DLL的delete
std::tr1:;shared_ptr<Investment> createInvestment()
{
	return std::tr1::shared_ptr<Investment>(new Stock);
}

Boost的tr1::shared_ptr特点:

  • 是原始指针的两倍大
  • 以动态分配内存作为簿记用途和删除器的专属数据
  • 以virtual形式调用删除器
  • 在多线程程序修改引用次数时有线程同步化(thread synchronization)的额外开销

Tips:

  • 好的接口不易被误用
  • 促进正确使用的方法包括接口一致性和与内置类型的行为兼容
  • 阻止误用的办法包括建立新类型、限制类型上的操作、束缚对象值、消除客户的资源管理责任
  • tr1::shared_ptr支持定制型删除器(custom deleter),这可以防范DLL问题,可被用来自动解除互斥锁(mutexes)等

条款19:设计class犹如设计type(Treat class design as type design)

定义一个新class时也就定义了一个新type。设计高效的类需要考虑以下问题:

  • 新type的对象应如何创建和销毁(第8章))
    • 影响构造函数和析构函数、内存分配函数和释放函数(operator new,operator new [],operator delete,operator delete [])
  • 对象的初始化和赋值应有什么差别(条款4)
    • 决定构造函数和赋值操作符的行为
  • 新type的对象如果被pass-by-value意味着什么
    • 由copy构造函数定义pass-by-value如何实现
  • 什么是新type的合法值
    • 有效的数值集决定了类必须维护的约束条件(invariants),
      • 进而决定了成员函数(特别是构造函数、析构函数、setter函数)的错误检查
    • 还影响函数抛出的异常和极少使用的函数异常明细列(exception specifications)
  • 新type需要配合某个继承图系(inheritance graph)吗
    • 继承既有的类,则受那些类束缚,尤其要考虑那些类的函数是否为虚函数
    • 被其他类继承,则影响析构函数等是否为virtual
  • 新type需要什么样的转换
    • 若允许类型T1隐式转换为类型T2,可可考虑:
      • 在T1类内写类型转换函数(operator T2)
      • 在T2类内些non-explicit-one-argument(可被单一实参调用)的构造函数
      • 若只允许explicit构造函数存在,就得写专门执行转换的函数,且没有类型转换操作符(type conversion operators)或non-explicit-one-argument构造函数
  • 什么样的操作符和函数对于此新type合理
    • 决定需要声明哪些函数,其中哪些是成员函数
  • 什么样的标准函数应驳回
    • 这些必须声明为private
  • 谁改取用新type的成员
    • 影响public、private、protected的选择
    • 影响友元类、友元函数、及其嵌套的设计
  • 什么是新type的未声明接口(undeclared interface)
    • 要考虑其对效率、异常安全性、资源运用的保证
  • 新type有多么一般化
    • 若要定义整个type家族,则应该定义新的class template
  • 是否真的需要新type
    • 若定义新的派生类就足够,则可能定义non-member函数或templates更好

Tips:

  • Class设计就是type设计,需要考虑以上所有问题

条款20:宁以pass-by-reference-to-const替换pass-by-value(Prefer pass-by-reference-to-cons to pass-by-value)

避免构造和析构

class Person {
public:
	Person();
	virtual ~Person();
	...
private:
	std::string name;
	std::string address;
};
class Student: public Person {
public:
	Student();
	~Student();
	...
private:
	std::string schoolName;
	std::string schoolAddress;
};
bool validateStudent(Student s);  //会调用六次构造函数和六次析构函数
bool validateStudent(const Student& s);  //效率提升很多

上述代码validateStudent函数中pass-by-value会调用六次构造函数和六次析构函数:

  • Student构造+Person构造+Student的2个string+Person的2个string
  • 析构同理

使用pass-by-reference可避免频繁构造和析构

避免对象切割

对象切割(slicing):派生类以值传递并被视为基类对象时,回调用基类的构造函数,而派生类的成分全无

class Window {
public:
	...
	std::string name() const;  //返回窗口名称
	virtual void display() const;  //显示窗口和其内容
};
class WindowWithScrollBars: public Window {
public:
	...
	virtual void display() const;
};

void printNameAndDisply(Window w)
{
	std::cout << w,name();
	w.display();
}
WindowWithScrollBars wwsb;
printNameAndDisply(wwsb);  //对象会被切割,因为参数w时Window对象,故调用的Window::display
void printNameAndDisply(const Window& w)  //不会被切割
{
	std::cout << w,name();
	w.display();
}

例外

  • 内置类型和STL的迭代器与函数对象采用pass by value往往效率更高,
  • 小型type不一定适合pass by value
    • 一旦需要复制指针的所指物,则copy构造函数可能成本很高
  • 即使小型对象的copy构造函数不昂贵,其效率也存在争议
    • 某些编译器对内置类型和自定义类型的态度截然不同,即使二者底层表示(underlying representation)相同
    • 如可能会把一个double放入缓存器,而只包含一个double的对象则不会
    • by reference则肯定把指针放入缓存器
  • 用户自定义类型的大小容易变化,因其内部实现可能改变,故不一定适合pass by value
    • 某些标准程序库实现版本中的string类型比其他版本大七倍

Tips:

  • 尽量以pass-by-reference-to-const替换pass-by-value。前者通常高效且能避免切割问题
  • 以上规则并不适用内置类型和STL的迭代和与函数对象,它们更适合pass-by-value

条款21:必须返回对象时,别妄想返回其reference(Don’t try to return a reference when you must return an object)

考虑有理数乘积:

class Rational {
public:
	Rational(int numerator = 0, int denominator = 1);
	...
private:
	int n, d;  //分子和分母
	friend const Rational operator* (const Rational& lhs, const Rational& rhs);
};

上述代码中操作符以by value的方式返回值,如果要返回reference则操作符必须自己创建新Rational对象,其途径有二:在stack或heap空间创建(反例)

//返回local对象的引用,但是local对象在离开函数时就销毁了
const Rational& operator* (const Rational& lhs, const Rational& rhs)
{
	Rational result(lhs.n * rhs.n, lhs.d * rhs.d);
	return result;
}

//难以对new创建的对象delete,尤其以下连乘的例子
const Rational& operator* (const Rational& lhs, const Rational& rhs)
{
	Rational* result = new Rational(lhs.n * rhs.n, lhs.d * rhs.d);
	return *result;
}
//无法取得引用背后的指针
Rational w, x, y, z;
w = x * y * z;  //operator*(operator*(x, y), z)

若使用static Rational避免调用构造函数,则会有如下问题:

//返回local对象的引用,但是local对象在离开函数时就销毁了
const Rational& operator* (const Rational& lhs, const Rational& rhs)
{
	static Rational result;
	result = ... ;
	return result;
}
bool operator==(const Rational& lhs, const Rational& rhs);
Rational a, b, c, d;
...
if ((a * b) == (c * d)) { ... }  //==总是为true
else { ... }  //,因两侧是同一个同一个stetic Rational对象的引用

必须返回新对象的函数的正确写法为:

inline const Rational& operator* (const Rational& lhs, const Rational& rhs)
{
	return Rational(lhs.n * rhs.n, lhs.d * rhs.d);
}

Tips:

  • 绝不要返回指针或引用指向local stack对象
  • 绝对不要返回引用指向heap-allocated对象
  • 绝对不要在有可能同时需要多个这样的对象时返回指针或引用指向local static对象

条款22:成员变量声明为private(Declare data members private)

这一条似乎没啥内容_

把成员变量声明为private的原因如下:

  • 接口的一致性:非public的成员函数只能通过函数访问,并且可以方便的设置读写权限
  • 封装
    • 把成员变量隐藏在函数接口背后,可以方便地更改实现方式
    • public成员变量修改后所有使用它们的客户码都会被破坏
    • protected成员变量修改后所有使用它们的派生类都会被破坏,其并不比public更具有封装性

Tips:

  • 切记把成员变量声明为private,这个赋予访问数据的一致性、可细微划分访问控制、保证约束条件、提供充分的实现弹性
  • protected并不比public更具封装性

条款23:宁以non-member、non-friend替换member函数(Prefer non-member non-friend functions to member functions)

考虑有个类表示网页浏览器:

class WebBrowser {
public:
	...
	void clearCache();
	void clearHistory();
	void removeCookies();
	void clearEverything();  //执行所有清除动作
	...
};

执行所有清除动作由两个方案:

  • WebBrowser 提供函数
  • 由non-member函数调用相应的member函数
    • 封装性更高,且包裹弹性(packaging flexibility)较大,编译相依度较低,是更好的方案
//WebBrowser 提供函数
class WebBrowser {
public:
	...
	void clearEverything();  //执行所有清除动作
	...
};
//由non-member函数调用相应的member函数
void clearBrowser(WebBrowser& wb)
{
	wb.clearCache();
	wb.clearHistory();
	wb.removeCookies();
}

两点注意事项:

  • 准确地说,封装性良好的是non-member non-friend函数,而非non-member函数
  • 一个类的non-member non-friend函数可以是可以是另一个类的member
    • 有些语言的函数必须定义在类内(如Eiffel,Java,C#),可以令clearBrowser成为某个工具类(utility class)的一个static member函数,而非WebBrowser的一部分或friend
    • 在C++中可让clearWebBrowser成为non-member函数且和WebBrowser位于同一命名空间
namespace WebBrowserStuff {
	Class WebBrowser { ... };
	void clearBrowser(WebBrowser& wb);
	...
}

命名空间能跨越多个源码文件而类不能,故可将同一命名空间下不同功能类型的函数放在不同的头文件:

标准程序库也不是一个庞大的单一头文件,而是有若干个头文件,每个头文件声明std的某些功能,这样可以使得用户只依赖所使用的一小部分系统

//头文件webbrowser.h,包含WebBrowser自身和核心功能
namespace WebBrowserStuff {
class WebBrowser { ... };
	...  //核心功能,如广泛使用的non-member函数
}
//头文件webbrowserbookmarks.h,
namespace WebBrowserStuff {
	...  ////与书签相关的函数
}
//头文件webbrowsercookies.h,
namespace WebBrowserStuff {
	...  //与cookie相关的函数
}

Tips:

  • 宁可拿non-member non-friend函数替换member函数,以增肌封装性、包裹弹性、功能扩展性

条款24:若所有参数皆需类型转换,请为此采用non-member函数(Declare non-member functions when type conversions should apply to all parameters)

考虑有理数类:

class Rational {
public:
	Rational(int numerator = 0,    //构造函数刻意不为explicit
					 int denominator = 1);  //允许int到Rational的隐式转换
	int numerator() const;  //分子和分母的访问函数
	int denominator() const;
private:
	...
};

若操作符*为Rational的成员函数:

class Rational {
public:
	...
	const Rational operator* (const Rational& rhs) const;
};
Rational oneHalf(1, 2);
Rational result = oneHalf * 2;  //正确,发生了隐式类型转换,根据int创建了Rational
result = oneHalf.operator*(2);  //但如果是explicit构造函数则错误

result = 2 * oneHalf;  //错误!
result = 2.operator*(oneHalf);  //错误!重写上式,错误一目了然

result = operator*(2, oneHalf);  //错误!本例不存在接受int和Rational的操作符*

只有参数位于参数列内,这个参数才能隐式类型转换

要支持混合运算,则可让操作符*成为non-member函数:

const Rational operator*(const Rational& lhs, const Rational& rhs)
{
	return Rational(lhs.numerator() * rhs.numerator(),
									lhs.denominator() * rhs.denominator());

member函数的反面是non-member函数,而非friend函数

从Objected-Oriented C++转换到Template C++且Rational是一个class template时,本条款需要考虑新的问题

Tips:

  • 若需要为某个函数的所有参数(包括this指针所指的那个隐喻参数)进行类型转换,那这个函数必须是non-member

条款25:考虑写出一个不抛异常的swap函数(Consider support for a non-throwing swap)

缺省的swap

缺省情况下swap动作可由标准程序库提供的swap算法完成:

namespace std {
	template<typename T>  //只要T支持copying即可实现swap
	void swap(T& a, T& b)
	{
		T temp(a);
		a = b;
		b = temp;
	}
}

特化的swap

缺省的swap涉及三个对象的复制,而pimpl手法(pointer to implementation)可避免这些复制:

置换两个Widget对象值只需要置换其pImpl指针;而缺省的swap会复制三个Widget,并且复制三个WidgetImpl对象

class Widget {
public:
	Widget(const Widget& rhs);
	Widget& operator=(const Widget& rhs)  //复制Widget时,就复制WidgetImpl对象
	{
		...
		*pImpl = *(ths.pImpl);
		...
	}
	...
private:
	WidgetImpl* pImpl;  //所指对象内涵Widget数据
};

将std::swap针对Widget特化可解决上述问题:

令Widget声明public swap成员函数做真正的置换工作(采用成员函数是为了取用private pImpl,non-member函数则不行),再把std::swap特化

class Widget {
public:
	...
	void swap(Widget& other)
	{
		using std::swap;  //这个声明有必要,稍后解释
		swap(pImpl, other.pImpl);  //真正做置换工作,
	}
	...
};
namespace std {
	template<>  //表示其是std::swap的全特化(total template specialization)版本
	void swap<Widget>(Widget& a, Widget& b)
	{
		a.swap(b);  //要置换WIdget就调用其swap成员函数
	}
}

上述代码与STL容器有一致性,因为所有STL容器也都提供有public swap成员函数和std::特化版本(以调用成员函数)

若Widget和WidgetImpl都是class template,可考虑把WidgetImpl内的数据类型参数化:

template<typename T>
class WidgetImpl { ... };
template<typename T>
calss Widget { ... };

此时特化std::swap会遇到问题:

//以下代码企图偏特化(partially specialize)一个function template(std::swap)
//但C++只允许对class template偏特化
//故无法通过编译(虽然少数编译器错误地通过编译)
namespace std {
	template<typename T>
	void swap<Widget<T>>(Widget<T>& a, Widget<T>& b)
	{ a.swap(b);}
}

//偏特化function template时,通常会添加重载版本
//但以下代码也不合法,因为std不能添加新的templates,这由C++彼岸准委员会决定
namespace std {
	template<typename T>
	void swap(Widget<T>& a, Widget<T>& b)  //注意swap之后没有<...>
	{ a.swap(b); }
}

解决方案:声明一个non-memebr swap以调用member swap,但不再将non-member swap声明为std::swap的特化版本或重载版本

任何代码如果要置换两个Widget对象而调用swap,则C++的名称查找法则(name lookup rules;更具体地说是argument-dependent lookup或Koeing lookup法则)会找到WidgetStuff内的Widget专属版本

namespace WidgetStuff {  //为简化,把Widget相关功能都放入WidgetStuff命名空间内
	...
	template<typename T>
	class Widget { ... };
	...
	template<typename T>
	void swap(Widget<T>& a, Widget<T>& b)  //non-member swap函数,不属于std命名空间
	{
		a.swap(b);
	}
}

若想要class专属版的swap在尽可能多的语境下被调用,则需呀在该class所在的命名空间内写一个non-member版本和一个std::特化版本,故应该为该class特化std::swap

若希望调用T专属版本,并且在该版本不存在的情况下调用std内的一般化版本,可实现如下:

C++的名称查找法则确保会找到global作用域或T所在的命名空间内的任何T专属的swap;若没有专属swap则using声明使得能够调用std::swap

template<typename T>
void doSomething(T& obj1, T& obj2)
{
	using std::swap;  //令std::swap在此函数内可用
	...
	swap)obj1, obj2);  //为T调用最佳swap版本
	...
}

std::swap(obj1, obj2);  //错误的方式!强迫编译器调用std::swap

使用swap的总结

swap的使用总结如下:

  1. 若缺省的swap的效率可接受,则无需做额外的事
  2. 若缺省的swap效率不足,则可考虑:
    1. 提供public swap成员函数,使其置换相应类型的两个对象值,且绝不抛出异常
    2. 在class或template所在的命名空间内提供一个non-member swap,并调用上述swap成员函数
    3. 若正在编写class(而非class template),则特化std::swap并使其调用swap成员函数
  3. 若调用swap,则需要包含using声明式,使std::swap在函数内可见,之后不加namespace直接调用swap

成员版swap绝不可抛出异常,其最好的应用是帮助class或class template提供强烈的异常安全性(exception-safety)保障

Tips:

  • 当std::效率不高时,提供一个swap成员函数,并确保其不抛出异常
  • 如果提供一个member swap,则要提供一个non-member swap调用前者。对于class(而非template),也最好特化std::swap
  • 调用swap时应声明 using std:;swap,之后不带命名空间修饰地调用swap
  • 为自定义类型进行std template全特化可以,但是不要再std内加入新东西`
posted @ 2024-05-02 13:49  Roanapur  阅读(76)  评论(0编辑  收藏  举报