条款18: 让接口容易被正确使用,不易被误用
本条款通过设计时间类Date来说明问题,外覆类型来区分天数Day、月份Month和年份Year,通过查阅网上的资料,练习Date类的相关知识,但是并没有用到该条款提到的,不知该如何使用???
令工厂函数返回一个智能指针(防止忘记删除底部对象),定制删除器。
本条款后半部分还缺少练习!!!
条款19:设计class犹如设计type
设计规范
(1)创建和销毁?—— 构造函数、析构函数、内存分配函数、释放函数
(2)初始化?赋值?——构造函数、赋值操作符
(3)传值?——如何实现
(4)type合法值?——-约束条件、错误检查、函数异常
(5)需要继承图系?——–哪些函数为virtual?
(6)类型转换?——-类型转换函数
(7) 操作符、函数?—–哪些是member函数
(8)哪个函数应该驳回?——声明为private
(9)未声明接口?—–约束条件
(10)一般化?——-class? class template?
(11)需要新type?—-derived class 可能通过定义non-member函数即可实现。
条款20 以pass-by-reference-to-const 替换 pass-by-value
缺省是by-value(副本)
reference一指针实现的
前者高效,并可避免切割问题(基类和派生类)
不适用于内置类型,以及STL的迭代器和函数对象,pass-by-value适当。
条款21 必须返回对象时,别妄想返回其reference
通过例子来进行说明
class Rational{
friend const Ratioal operator *
(const Rational &lhs,conat Rational &rhs);
public:
Rational(int numerator = 0,int denominator = 1 );
......
private:
int n,d;
}
引用的另一个名称呢?
const Ratioal**&** operator *
(const Rational &lhs,conat Rational &rhs)
{
Ratiaonal Result(lhs.n*rhs.n,lhs.d*rhs.d); //Bad
return Result;
}
(1)stack建对象
Result在函数退出前被销毁(local 对象)
返回引用指向local对象is bad
(2)heap建对象
const Ratioal& operator *
(const Rational &lhs,conat Rational &rhs)
{
Ratiaonal*** Result= new Rational(lhs.n*rhs.n,lhs.d*rhs.d); //worse
return*** Result;
}
哪里实施delete?(new和delete成对存在)
(3)函数内部静态(static)对象
const Ratioal**&** operator *
(const Rational &lhs,conat Rational &rhs)
{
static Ratiaonal Result ; //Bad
result=...
.....
return Result;
}
Rational a,b,c,d;
bool flag=(a*b) == (c*d);//flag总是true
static Rational 对象的“现值”总是一样。
const Ratioal**&** operator *
(const Rational &lhs,conat Rational &rhs)
{
return Ratiaonal(lhs.n*rhs.n,lhs.d*rhs.d); //good
}