您的位置:首页 > 编程语言 > C语言/C++

Effective C++第三遍

2016-03-15 18:16 471 查看
试图调用private的copy或赋值函数是编译期错误,而调用没有具体定义的函数则是连接期错误。

以对象管理资源;智能指针RAII(资源获取立即初始化)后都是对象,但有时候,比如(API的)函数参数要求的是原始资源的指针,这时候有两种方法:

1、显式转换:智能指针提供了get成员函数可以直接获取原始资源:

std::tr1::shared_ptr<Investment> pInv(createInvestment());

int daysHeld(const Investment* pi);

int days=daysHeld(pInv.get());

2、隐式转换:智能指针重载了指针取值操作符(->和*);或者提供一个隐式转换函数。

内置类型参数通常被设计为pass-by-value,STL的迭代器和函数对象通常也被设计为pass-by-value。而其他很多class通常被设计为pass-by-reference,因为pass-by-value参数采用复制,会有很多构造和析构的成本。

public意味着不封装,而不封装几乎意味着不可改变,因为会涉及到(破坏)太多的用户代码。

能够访问private变量的只有成员函数和friend函数。因此使用非成员函数,有更好的封装性。

C++比较自然的做法是设计为非成员函数,然后位于目标class所在的同一个namespace内。

只有当参数位于参数列内时,才允许隐式转换,所以r*2正确(调用r.operater*(2),这里2被隐式转换为有理数),但2*r错误(因为没有相应函数可调用);一个可行的方法是让*成为一个non-member函数:const Rational operator*(const Rational& lhs,const Rational& rhs)。这样2*r也正确,因为2和r都位于参数列内,都被隐式转换为有理数Rational。

pimpl手法:以指针指向一个对象,内含真正数据。例如针对widget设计:

首先定义一个类widgetImpl包含实际的数据,然后类widget包含一个私有成员是widgetImpl的指针widgetImpl*。

全特化的写法:

namespace std{
template<>
void swap<widget>(widget& a,widget& b){
a.swap(b);
}
}


客户可以全特化std内的template,但不可以添加新的Template(或class或function或其他任何东西)到std里头。

因此,为使swap在尽可能多的语境下被调用,需要同时在目标class所在命名空间内写一个non-member版本及一个std::swap特化版本。

降低文件间的编译依存关系:

如果引用或指针能完全任务,就不要使用对象(这样不需要知道对象的大小,而只需要指针);

尽量以class声明式替换class定义式;

为声明式和定义式提供不同的头文件。

public继承:is-a ,适用于base classes身上的每一件事情一定也适用于派生类。现实中的is-a未必全适合这里的公有继承。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: