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*。
全特化的写法:
客户可以全特化std内的template,但不可以添加新的Template(或class或function或其他任何东西)到std里头。
因此,为使swap在尽可能多的语境下被调用,需要同时在目标class所在命名空间内写一个non-member版本及一个std::swap特化版本。
降低文件间的编译依存关系:
如果引用或指针能完全任务,就不要使用对象(这样不需要知道对象的大小,而只需要指针);
尽量以class声明式替换class定义式;
为声明式和定义式提供不同的头文件。
public继承:is-a ,适用于base classes身上的每一件事情一定也适用于派生类。现实中的is-a未必全适合这里的公有继承。
以对象管理资源;智能指针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未必全适合这里的公有继承。
相关文章推荐
- C++运算符重载:怎样定义函数才“符合常理”
- c/c++内存机制(一)(转)
- C++中关于函数的引用
- c++ 内存五区理解
- iOSDay08之C语言内存管理
- C++:为什么有容器与迭代器
- C语言找出数组中的特定元素的算法解析
- 《C++ Primer》第四版 第13章 复制控制
- C++纯虚函数
- C语言将数组中元素的数排序输出的相关问题解决
- C语言中的强符号与弱符号
- C++:冒号与双冒号用法
- C++
- c++ 走向高级之日积月累
- C++中this指针的用法及介绍
- C/C++问题汇总
- C++ pair
- C++之多态性与虚函数
- C语言输出旋转后数组中的最小数元素的算法原理与实例
- 关于C++的虚函数和多态性