C++ 转型动作 尽量避免 以及 那些意想不到的威胁
看完EffectiveC++的关于转型的章节,顿时认为周围的代码都处在悬崖边上~~
C的旧式转型:inta
= 10; double b = (double)a;
对于C++的四种转型函数,
const_cast去掉对象的常量性(仅仅此一个操作符有此功能!)
dynamic_cast一般用于继承体系中某对象的归属,耗费较大
reinterpret_cast低级转型,差点儿不用。
static_cast强迫隐式转换int->double,
void * ->typed *
在类的expecilit构造函数中,比如:
class Widget
{
public:
explicit: Widget(int size);
....
};
旧式转型有时也用,比方在,对于一个须要Widget对象作为參数的函数中,比如:voidf(
Widget &w ); 调用时,使用旧式转型f(Widget(15));
转型并不不过只告诉编译器将该对象视为什么类型,编译器会做出一些动作来完毕转型,比如:
将int转化为double,在完毕浮点数的乘法的时候,显然是要产生不同的目标码的。
以下的样例可能你每天都在用,那就是"基类的指针指向子类的对象",比如:
class Based { ... };
class Derived: public Based{ ... };
Derived de;
Based *pb = &de; //基类的指针指向子类的对象。
在实际。这两个指针的值可能不一样,也许是在当前对象上加了一个偏移量,这也就意味着,同一个Derived对象可能有两个地址,基类类型的指针与派生类类型的指针可能是不一样的。当然这依赖于内存布局(啊哦,不可移植性~~)
一个接着一个,看似平时非常习惯的东西,却一个一个的有着潜在的威胁:
class Parent
{
public:
virtual void Func() {...}
...;
};
class Child : public Parent
{
public:
virtual void Func()
{
static_cast<Parent>(*this).Func();
...; //子类自己的操作。
}
};
在子类中调用父类定义的函数。这是再寻常只是的事了。
但是以下的代码却非常危急:
上面这段简单的代码,子类的Func函数首先调用父类的,再做子类的操作。但是,这里的static_cast得到的是一个暂时副本,那么也就是说你的Func函数假设试图改动子类的成员,那么父类的部分是再暂时副本上做的,也就是说,你可能会得到一个“父类没有改动,子类改动了的”病态的对象。太可怕了,不是要尽量使用static_cast。要实现上述功能,直接指定作用域就可以,即使用Parent::Fun()函数来调用。
PS:尽量少用转型。
对于dynamic_cast,那就更应当敬而远之了,除了运行速度非常慢,尤事实上在多重继承或是多层继承中,简单的“类名搜索”成本就非常高了。
那么我们就必须知道,什么时候我们必须使用dynamic_cast呢?
如今你仅仅有“指向base”的指针或者引用,可是你须要的是对一个derived对象运行derivedclass的操作函数,你仅仅能靠它们来处理对象。这里全然能够避免使用dynamic_cast,方法为:
1不要忘了虚函数。。详细定义參见:
2不要忘了智能指针,使用容器存放指向derived类的智能指针,这样变消除了“使用base接口来处理对象”的必要性,关于智能指针,參见智能指针
请记得:
在注重效率的应用中,对转型动作“敬而远之”吧!
假设必须转型,那么记得封装在函数里面,不要让这些“恼人”的代码干扰用户代码。
使用新式转型,各司其职,不易出错,比較easy在源码中识别出来。
假设你认为你必须使用转型,请再想想,迈出这危急的一步的代价~~~
C++ 转型动作 尽量避免 以及 那些意想不到的威胁的更多相关文章
- Effective C++ -----条款27:尽量少做转型动作
如果可以,尽量避免转型,特别是在注重效率的代码中避免dynamic_casts.如果有个设计需要转型动作,试着发展无需转型的替代设计. 如果转型是必要的,试着将它隐藏于某个函数背后.客户随后可以调用该 ...
- [Effective C++ --027]尽量少做转型动作
引言 ...
- 读书笔记_Effective_C++_条款二十七:尽量少做转型动作
有关转型的几种做法,已经在早些的博客中写过了.这里先简单回顾一下,再讲一讲effective中对之更深入的阐述. 转型可以按风格可以分成C风格转型和C++风格转型两大类,C风格转型很容易看到,因为我们 ...
- 条款27:尽量少做转型动作(Minimize casting)
NOTE : 1.如果可以,尽量避免转型,特别是在注重效率的代码中避免dynamic_casts. 如果有个设计需要转型动作,试着发展无需转型的替代设计. 2.如果转型是必须要的,试着将它隐藏于某个函 ...
- [EffectiveC++]item27:尽量少做转型动作
- 代码优化—From <effective C++>
1.尽可能的延后变量定义式的出现时间 不止应该延后变量的定义,直到非得使用该变量的前一刻为止,甚至应该尝试延后这份定义直到能够给它初值实参为止. 如果这样不仅能够避免构造和析构非必要对象,还可以避免无 ...
- Effective C++笔记(一)——条款26-29
条款26:尽可能延后变量定义式的出现时间 为何要尽量延后? 当程序中途跳出而导致变量未被使用,但是必须进行构造和析构. 最佳初始化变量 直接在构造时指定初值比构造之后再赋值效率高(条款4) ... s ...
- Effective C++ 条款27
尽量少做转型动作 尽量少做转型动作有什么目的?非常明显无非就是提高程序的稳定性.提高程序的运行效率. 那么.有哪些转型方式?每种方式都有什么弱点? 这是我们本节学习的重点. C++有四种转型: con ...
- Effective C++:规定27:尽量少做动作的过渡
(一个)C风格遗留转换: (T)expression T(expression) (二)C++提供四种新式转型: (1)const_cast<T>(expression):去除表达式的常量 ...
随机推荐
- 【LOJ】#2289. 「THUWC 2017」在美妙的数学王国中畅游
题解 我们发现,题目告诉我们这个东西就是一个lct 首先,如果只有3,问题就非常简单了,我们算出所有a的总和,所有b的总和就好了 要是1和2也是多项式就好了--其实可以!也就是下面泰勒展开的用处,我们 ...
- jenkins升级
1丶查看war包所在的目录 #rpm -ql jenkins /etc/init.d/jenkins //jenkins的启动文件 /etc/logrotate.d/jenkins //日志切割 /e ...
- linux 下rocketmq安装
一.解压mq(/data下)tar -zxvf Rocketmq-3.5.8.tar.gz 二.修改配置文件vi /etc/profileexport rocketmq=/data/alibaba-r ...
- react篇章-React State(状态)-组件都是真正隔离的
<!DOCTYPE html> <html> <head> <meta charset="UTF-8" /> <title&g ...
- DNS隧道工具iodine
DNS隧道工具iodine 在受限制的网络中,如果DNS请求没有被限制,就可以通过DNS请求建立隧道而突破网络限制.iodine是Kali Linux提供的一款DNS隧道工具.该工具分为服务器端i ...
- Form与ModelForm-下拉框或者多选注意//及字段补充
一.Form 设计一张普通model表: class UserInfo(models.Model): name = models.CharField(verbose_name='员工姓名', max_ ...
- 【BZOJ 3036】 3036: 绿豆蛙的归宿 (概率DP)
3036: 绿豆蛙的归宿 Time Limit: 2 Sec Memory Limit: 128 MBSubmit: 491 Solved: 354 Description 随着新版百度空间的下线 ...
- 希尔排序之Java实现
希尔排序之Java实现 一.方法一 package cn.com.zfc.lesson21.sort; /** * * @title ShellSort * @describe 希尔排序 1959 年 ...
- Luogu 4492 [HAOI2018]苹果树 组合数
https://www.luogu.org/problemnew/show/P4492 找每个编号的点的父边的贡献,组合数和阶乘就能算了. 我考场上怎么就是没想到呢. 调了好久好久好久好久调不出来,样 ...
- [BZOJ 4870] 组合数问题
Link: 传送门 Solution: 组合数的式子都可以先想想能不能递推,写出来就是: $\sum C_{n*k}^{i*k+r}=\sum C_{n*k-1}^{i*k+r}+\sum C_{n* ...