Effective C++:规定27:尽量少做动作的过渡
(一个)C风格遗留转换:
(T)expression
T(expression)
(二)C++提供四种新式转型:
(1)const_cast<T>(expression):去除表达式的常量性。是C++中唯一能做此操作的转型操作符。
(2)dynamic_cast<T>(expression):主要用来运行“安全向下转型”,即用来决定某对象是否归属继承体系中的某个类型。它是唯一无法由旧式语法运行的动作,也是唯一可能耗费重大运行成本的转型动作。
(3)reinterpret_cast<T>(expression):试图运行低级转型(比如将一个指向整数变量的指针转换为整数),实际动作及结果可能与编译器相关,因而减少了可移植性。
(4)static_cast<T>(expression):隐式转换。注意尽管能够使非const类型转换为const。但反之不能(这个仅仅能由const_cast来完毕)。
(三)新式转型比較受欢迎,原因:
1,非常easy在代码中被辨别出来。
2,各转型动作的目标愈窄化。编译器愈可能诊断出错误的运用。
(四)唯一使用旧式转型的时机是当调用一个explicit构造函数将一个对象传递给一个函数时:
class Widget{
public:
explicit Widget(int size);
...
};
void doSomeWork(const Widget &w);
doSomeWork(Widget(15));//函数风格的转型动作创建一个Widget
doSomeWork(static_cast<Widget>(15));//c++风格的转型动作创建一个Widget对象
(五)先看以下这个代码:
class Base{...};
class Derived : public Base{...};
Derived d;
Base* pb = &d;//隐喻的将derived*转换成Base*
在这里表明:单一的对象可能拥有一个以上的地址。这里我们仅仅是建立一个base
class指针指向一个derived class对象,但有时候上述的两个指针值并不同样。这样的情况下会有个偏移量offset在执行期被施行于Derived*指针身上,用于取得正确的Base*指针值。
实际上一旦使用多重继承,这事差点儿一直发生.即使在单一继承中也可能发生.恐怖!为什么会发生这种事情呢?
由于对象的布局方式和它们的地址计算发式随着编译器的不同而不同,这就以为着写出"依据对象怎样布局"而写出的转型代码在某一平台上行得通,在其他平台上则不一定。
(六)
还有一件关于转型的有趣事情是:我们非常easy写出某些似是而非的代码(其它语言中或许是对的)。
比如SpecialWindow的onResize被要求首先调用Window的onResize。以下是看起来对,实际上错:
class Window {
public:
virtual void onResize() {...}
};
class SpecialWinddow : public Window {
public:
virtual void onResize() {
static_cast<Window>(*this).onResize();//将*this转换成Window,然后调用其onResize;这样不行!
}
};
这里会发生什么问题呢?static_cast<Window>(*this)这个转型动作并非如你想象的那样得到当前对象的基类对象部分,事实上编译器为你产生了的是基类对象的副本。
我运行的onResize方法压根就没有运行到基类对象上。而是运行在它的副本上。
可是SpecialWindow的专属onResize却运行在子类对象上。使得这个对象处于一种"伤残状态“。
解决方法:
class SpecialWinddow:public Window{
public:
virtual void onResize(){
Window::onResize();//调用Window::onResize作用于*this身上
...
}
};
(七)
首先。dynamic_cast的运行速度相当慢。
之所以要用dynamic_cast,一般是由于你想在一个你认定为derived
class对象身上运行derived class操作函数。但你的手上仅仅有一个“指向base”的pointer或者reference。你仅仅能靠他们来处理对象。
可是我们应该尽量避免使用dynamic_cast,有两种替代方法!
第一种方法:
使用容器并在当中存储直接指向derived
class对象的指针(一般是智能指针)如此便消除了“通过base class接口处理对象”的须要。
不要像以下这样用dynamic_cast:
class Window{...};
class SpecialWindow:public Window {
public:
void blink();
};
typedef std::vector<std::tr1::shared_ptr<Window>> VPW;
VPW winPtrs;
for(VPW::iterator iter = winPtrs.begin(); iter != vinPtrs.end(); ++iter) {
if(SpecialWindow* psw = dynamic_cast<SpecialWindow*>(iter->get()))
psw->blink();
}
要改成这样:
typedef std::vector<std::tr1::shared_ptr<SpecialWindow>> VSPW;
VSPW winPtrs;
for(VSPW::iterator iter = winPtrs.begin(); iter != vinPtrs.end(); ++iter) {
(*iter)->blink();
}
另外一种方法:
能够通过base class接口处理“全部可能之各种window派生类”,那就是在base class 里提供virtual函数做你想对各个Window派生类做的事。
class Window {
public:
virtual void blink() { }
};
class SpecialWindow : public Window {
public:
virtual void blink() {...};
};
typedef std::vector<std::tr1::shared_ptr<Window>> VPW;
VPW winPtrs;
for(VPW::iterator iter = winPtrs.begin(); iter != vinPtrs.end(); ++iter) {
(*iter)->blink();
}
(八)
绝对必须拒绝的是所谓的“连串(cascading)dynamic_casts”:
typedef std::vector<std::tr1::shared_ptr<Window>> VPW;
VPW winPtrs;
for (VPW::iterator iter = winPtrs.begin(); iter != winPtrs.end(); ++iter)
{
if (SpecialWindow1 * psw1 = dynamic_cast<SpecialWindow1*>(iter->get())) {...}
if (SpecialWindow2 * psw1 = dynamic_cast<SpecialWindow2*>(iter->get())) {...}
if (SpecialWindow3 * psw1 = dynamic_cast<SpecialWindow3*>(iter->get())) {...}
}
这种代码应该总是以某些“基于virtual函数调用”的东西取而代之。
优良的c++代码非常少使用转型,我们应该尽可能隔离转型动作。一般是把它隐藏在某个函数内,函数的接口会保护调用者不受函数内部不论什么肮脏龌龊的动作的影响。
请记住:
(1)假设能够,尽量避免转型,特别是在注重下来的代码中避免dynamic_cast.假设有个设计须要转型动作,试着发展无需转型的替代设计。
(2)假设转型是必要的,试着将它隐藏于某个函数背后.客户随后能够调用该函数,而不需将转型放进它们自己的代码内。
(3)宁可使用C++-style(新式)转型,不要使用旧式转型.前者非常easy辨识出来,并且也比較有着分门别类的职掌。
版权声明:本文博客原创文章,博客,未经同意,不得转载。
Effective C++:规定27:尽量少做动作的过渡的更多相关文章
- 读书笔记 effective c++ Item 27 尽量少使用转型(casting)
C++设计的规则是用来保证使类型相关的错误不再可能出现.理论上来说,如果你的程序能够很干净的通过编译,它就不会尝试在任何对象上执行任何不安全或无意义的操作.这个保证很有价值,不要轻易放弃它. 不幸的是 ...
- 读书笔记_Effective_C++_条款二十七:尽量少做转型动作
有关转型的几种做法,已经在早些的博客中写过了.这里先简单回顾一下,再讲一讲effective中对之更深入的阐述. 转型可以按风格可以分成C风格转型和C++风格转型两大类,C风格转型很容易看到,因为我们 ...
- Effective C++ -----条款27:尽量少做转型动作
如果可以,尽量避免转型,特别是在注重效率的代码中避免dynamic_casts.如果有个设计需要转型动作,试着发展无需转型的替代设计. 如果转型是必要的,试着将它隐藏于某个函数背后.客户随后可以调用该 ...
- [Effective C++ --027]尽量少做转型动作
引言 ...
- 条款27:尽量少做转型动作(Minimize casting)
NOTE : 1.如果可以,尽量避免转型,特别是在注重效率的代码中避免dynamic_casts. 如果有个设计需要转型动作,试着发展无需转型的替代设计. 2.如果转型是必须要的,试着将它隐藏于某个函 ...
- [EffectiveC++]item27:尽量少做转型动作
- Effective C++:条款27——条款
条款27:尽量少做转型动作 单一对象可能拥有一个以上的地址!
- Effective C++ 条款27
尽量少做转型动作 尽量少做转型动作有什么目的?非常明显无非就是提高程序的稳定性.提高程序的运行效率. 那么.有哪些转型方式?每种方式都有什么弱点? 这是我们本节学习的重点. C++有四种转型: con ...
- Effective Objective-C 2.0 — 第二条:类的头文件中尽量少引入其他头文件
第二条:类的头文件中尽量少引入其他头文件 使用向前声明(forward declaring) @class EOCEmployer 1, 将引入头文件的实际尽量延后,只在确有需要时才引入,这样就可以减 ...
随机推荐
- 用VS2005编译生成Lua库文件和解释器
TMD,本来很简单的东西,网上说的乱七八糟,说的也不明白,大家抄来抄去,估计都不自己实践的..花了半个下午研究了一下,总结一下. 1)下载lua工程文件,地址为http://www.lua.org/f ...
- SVN中正确的add操作和delete操作
add操作: delete操作:
- pygame系列_游戏窗口显示策略
在这篇blog中,我将给出一个demo演示: 当我们按下键盘的‘f’键的时候,演示的窗口会切换到全屏显示和默认显示两种显示模式 并且在后台我们可以看到相关的信息输出: 上面给出了一个简单的例子,当然在 ...
- Redis 的性能幻想与残酷现实(转)
2011 年,当初选择 Redis 作为主要的内存数据存储,主要吸引我的是它提供多样的基础数据结构可以很方便的实现业务需求.另一方面又比较担心它的性能是否足以支撑,毕竟当时 Redis 还属于比较新的 ...
- (九)通过几段代码,理清angularJS中的$injector、$rootScope和$scope的概念和关联关系
$injector.$rootScope和$scope是angularJS框架中比較重要的东西,理清它们之间的关系,对我们兴许学习和理解angularJS框架都很实用. 1.$injector事实上是 ...
- 集成框架 javaweb开发平台ssmy_m(生成代码) java struts2 mybatis spring maven jquery
网页地址 http://blog.csdn.net/lpy3654321/article/details/31841573 项目设想,在项目开发中,我们的开发者大多数时间都在反复开发 相同的keywo ...
- 勾股数组及其应用uva106
勾股数组 设三元组(a,b,c)满足a^2 + b^2 = c^2的勾股数组,那么是否存在无穷多个勾股数组呢, 答案是肯定的,将三元组乘以d,可以得到新的三元组(da,db,dc) 即(da)^2 + ...
- 将ACCESS数据库迁移到SQLSERVER数据库
原文:将ACCESS数据库迁移到SQLSERVER数据库 将ACCESS数据库迁移到SQLSERVER数据库 ACCESS2000文件 用ACCESS2007打开,并迁移到SQLSERVER2005里 ...
- 一个简单的java僵局演示示例
在实际编程,为了避免死锁情况,但是,让你写一个有用的程序死锁似几乎不要太简单(种面试题),下面是一个简单的死锁样例. 线程的同步化可能会造成死锁,死锁发生在两个线程相互持有对方正在等待的东西(实际是两 ...
- 03.Msbuild
MSBuild的深入认识 分类: 专题开发 自动化 2009-01-20 11:56 5711人阅读 评论(1) 收藏 举报 任务引擎脚本工作扩展build 最近在从事自动构造工作的过程中,对MSBu ...