effective c++ 笔记 (41-44)
//---------------------------15/04/25----------------------------
//#41 了解隐式接口和编译期多态
{
// 1:面向对象编程总是以显示接口和运行期多态解决问题:
void doProcessing(Widget& w)
{
&& w != someNastyWidget)
{
Widget temp(w);
temp.normalize();
temp.swap(w);
}
}
/* 1>w的类型被声明为Widget,所以w必须支持Widget接口,我们可以从源码中找出这个接口,看看
时什么样子的,这种接口称之为显式接口。
2>对于virtual函数,w将在运行期表现出多态。
2:Template的世界与面向对象不同,隐式接口和编译期多态变得很重要: */
template<typname T>
void doProcessing(T& w)
{
&& w != someNastyWidget)
{
T temp(w);
temp.normalize();
temp.swap(w);
}
}
/* 1>w必须支持哪一种接口是由执行于w身上的操作来决定的,比如上面看起来w必须支持size,
normalize,swap,copy构造函数,不等于比较等接口。这些便是隐式接口。
2>凡是涉及w的任何函数调用,有可能造成template具现化行为,这样的行为发生在编译期。
”以不同的template参数具现化function template“
会导致调用不同的函数,这就是编译
期多态。
3:再探隐式接口:
1>T并不需要提供一个名为size的成员函数,因为T有可能从base class继承而得。
2>T并不需要支持operator!=,因为有可能有个operator!=函数,接受类型为X的对象和类型
为Y的对象,而T可以被转换为X,someNastyWidget可以被转换为Y。
*/
}
//#42 了解typename的双重意义
{
/* 1:在声明template的时候,class和typename完全一样。
template<class T> 等价于 template<typename T>
2:当需要用到T中的类型时,需要明确指定是一个类型,也就是在前面加上一个typename:
typename T::const_iterator iter;
如果不指定的话,编译器时无法知道这是一个类型还是一个成员变量的,下面的语句可以造成歧义:
T::const_iterator* pIter;
到底是声明一个指针变量还是拿T中的const_iterator
和 pIter相乘呢?编译器不知道!
3:不允许使用typename的地方:
1>typename T t; 这样是错误的,typename知识用来验证嵌套从属类型名称的,前面那句声明
编译器是很明确知道的,不需要加typename。
2>typename不能出现在base class list
和 member.init.list: */
template<typename T>
class Derived: public Base<T>::Nested
//不需要typename
编译器知道的
{
public:
explicit Derived(int x) : Base<T>::Nested(x)
//不需要使用typename
{
}
}
}
//#43 学习处理模版化基类内的名称
{
/* 如果基类是一个模版类,子类继承这样的基类时,如果想要调用基类的函数必须使用以下三种方法之一,否则
无法通过编译,因为基类是有可能特化的,所以编译器无法确定基类中是否真的存在你想调用的函数。比如你想
调用基类的sendClear:
1>使用this来调用:
this->sendClear();
2>使用using声明式:
using Base<T>::sendClear;
sendClear();
3>指明被调用的函数位于base中(不推荐使用这种方法)
Base<T>::sendClear()
不推荐的理由是,这种调用没有多态可言,如果sendClear是呈现多态,那么这里只会调用基类的函数。
但是换个思维来想,如果是virtual函数,那么子类中应该有相应实现吧。就不用别的操作,直接调用了。
(也就是一开始就通过编译了,不会考虑着三种方法了)
*/
}
//#44 将与参数无关的代码抽离templates
{
/* 1:使用template可以节省时间和避免代码重复,然而,看起来简洁的代码有可能会使代码的二进制码重复
而造成代码膨胀。
2:如何优化:
1>抽离相同的东西放入模版基类中。比如抽离相同的函数,相同的成员变量等。
2>当有需要时,可以给基类配置一个成员变量,抽离的函数
需要使用到的 一个指向子类中数据
的指针。
3:虽然做了这么多优化,但是需要付出一定的代码,简单来说就是执行速度。
这样的代码局部性不强,所以高速缓存很有可能无法命中。导致效率下降。
4:还有导致代码膨胀的情况:
1>int和long在很多平台上有相同的二进制表示,所以list<int>,list<long>实现的版本的二进制
表示是一样的,但是却实现了两份一样的代码。
2>所有的指针,比如list<const int*>, list<int>, list<Aclass*>这些底层实现应该唯一。
可以通过特化list<T*>来防止膨胀。
5:第一情况造成的膨胀应该考虑空间和时间,从中选择一个。
后面两种膨胀的情况时可以避免的。
*/
}
effective c++ 笔记 (41-44)的更多相关文章
- [Effective JavaScript 笔记]第44条:使用null原型以防止原型污染
第43条中讲到的就算是用了Object的直接实例,也无法完全避免,Object.prototype对象修改,造成的原型污染.防止原型污染最简单的方式之一就是不使用原型.在ES5之前,并没有标准的方式创 ...
- Effective Java笔记一 创建和销毁对象
Effective Java笔记一 创建和销毁对象 第1条 考虑用静态工厂方法代替构造器 第2条 遇到多个构造器参数时要考虑用构建器 第3条 用私有构造器或者枚举类型强化Singleton属性 第4条 ...
- [Effective JavaScript 笔记] 第4条:原始类型优于封闭对象
js有5种原始值类型:布尔值.数字.字符串.null和undefined. 用typeof检测一下: typeof true; //"boolean" typeof 2; //&q ...
- [Effective JavaScript 笔记] 第5条:避免对混合类型使用==运算符
“1.0e0”=={valueOf:function(){return true;}} 是值是多少? 这两个完全不同的值使用==运算符是相等的.为什么呢?请看<[Effective JavaSc ...
- [Effective JavaScript 笔记]第3章:使用函数--个人总结
前言 这一章把平时会用到,但不会深究的知识点,分开细化地讲解了.里面很多内容在高3等基础内容里,也有很多讲到.但由于本身书籍的篇幅较大,很容易忽视对应的小知识点.这章里的许多小提示都很有帮助,特别是在 ...
- [Effective JavaScript 笔记]第27条:使用闭包而不是字符串来封装代码
函数是一种将代码作为数据结构存储的便利方式,代码之后可以被执行.这使得富有表现力的高阶函数抽象如map和forEach成为可能.它也是js异步I/O方法的核心.与此同时,也可以将代码表示为字符串的形式 ...
- [Effective JavaScript 笔记]第28条:不要信赖函数对象的toString方法
js函数有一个非凡的特性,即将其源代码重现为字符串的能力. (function(x){ return x+1 }).toString();//"function (x){ return x+ ...
- java effective 读书笔记
java effective 读书笔记 []创建和销毁对象 静态工厂方法 就是“封装了底层 暴露出一个访问接口 ” 门面模式 多参数时 用构建器,就是用个内部类 再让内部类提供构造好的对象 枚举 si ...
- Effective STL 笔记 -- Item 6 ~ 7: Container and Object Pointer
Effective STL 笔记 – Item 6 ~ 7: Container and Object Pointer 中间两次笔记被删掉了,简单补一下: Item 3 中提到如果将对象直接放入容器中 ...
- Item 5:那些被C++默默地声明和调用的函数 Effective C++笔记
Item 5: Know what functions C++ silently writes and calls 在C++中,编译器会自己主动生成一些你没有显式定义的函数,它们包含:构造函数.析构函 ...
随机推荐
- 灰度发布:灰度很简单,发布很复杂&灰度发布(灰度法则)的6点认识
什么是灰度发布,其要点有哪些? 最近跟几个聊的来的同行来了一次说聚就聚的晚餐,聊了一下最近的工作情况如何以及未来规划等等,酒足饭饱后我们聊了一个话题“灰度发布”. 因为笔者所负责的产品还没有达到他们产 ...
- MySQL crash-safe replication(1)
MySQL 5.6 对复制功能提供了新特性:slave 支持 crash-safe,可以解决之前版本中系统异常断电可能导致的 SQL thread 信息不准确的问题. 原文:Enabling cras ...
- 【转】Linux思维导图
[原文]https://www.toutiao.com/i6591690511763898888/ 1.Linux学习路径: 2.Linux桌面介绍: 3.FHS(文件系统目录标准): 4.Linux ...
- 【require.js】模块化开发
一.Require.js及AMD Require.js:是一个非常小巧的JavaScript模块载入框架,是AMD规范最好的实现者之一. AMD(Asynchronous Module Definit ...
- 【Nginx】开启 gzip和缓存
Nginx 开启 gzip和缓存 时间:2016-09-23 16:42:37 nginx 是一个高性能的 Web 服务器,之前也写过一些关于 nginx 的文章.为了提高博客的响应速度,可以从设置 ...
- 团队作业——Alpha冲刺 9/12
团队作业--Alpha冲刺 冲刺任务安排 杨光海天 今日任务:修复编辑界面与弹窗界面合并中出现的BUG 明日任务:希望完成编辑界面所有接口交互的功能 郭剑南 今日任务:优化图像预处理所有功能的函数代码 ...
- 13.3SolrCloud集群使用手册之Zookeeper指令
转载请出自出处:http://www.cnblogs.com/hd3013779515/ 1.upconfig java -classpath .:/home/solr/cloud/lib/* org ...
- BZOJ1121:[POI2008]激光发射器SZK(乱搞)
Description 多边形相邻边垂直,边长为整数,边平行坐标轴.要在多边形的点上放一些激光发射器和接收器.满足下列要求: 1发射器和接收器不能放置在同一点: 2发射器发出激光可以沿壁反射,最终到达 ...
- 随手练——P1141 01迷宫
1.暴力版 本质上就是求连通块数量,那么DFS或者BFS都行,暴力跑. 写完发现题目比较特殊,m次提问,那每次都暴力搜,肯定是要跑死了. #include <iostream> #incl ...
- Linux下RPM包的安装
Linux下RPM包安装 二进制包(RPM包.系统默认包) RPM安装 rpm -ivh 包全名(查询依赖网址:http://www.rpmfind.net) -i(install):安装 -v(ve ...