//---------------------------15/04/24----------------------------

//#35   考虑virtual函数以外的其他选择

{

/*

1:通常情况下对于有变动的东西,我们都会设计成virtual函数,然后由子类来继承,并重新实现,

但是,由于这个惯性思维,成了弱点。因为我们这样就没有想过别的实现方法。

2:用Non-Virtual Interface来代替virtual函数。

1>把virtual函数设计成private,并使用别的public函数调用。这里有一个很关键的一点:

private virtual函数能实现多态么?答案是肯定的。尽情地重定义吧。

2>在调用virtual函数的前后可以加上一些自己的控制,比如上锁,记录等等一切你想做的。

3>其实也不一定必须是private的,自己看情况定义吧。

3:使用Strategy模式:

1>当你需要virtual函数时,可以考虑存储一个仿函数对象,然后在一个public函数中调用这个

仿函数。

2>这个仿函数内部的操作不应该依赖于类的内部细节,不然还是乖乖用virtual函数来的好。

3>这时你会发现,可以动态地改变要调用的函数,比如一个计算血量值的函数,可以一开始调用

正常版本的,当角色中毒什么的,就换一个计算血量的函数。哇!cool。

*/

}

//#36   绝不重新定义继承而来的non-virtual函数

{

/*

前提:D public继承自 B

先看两个准则(前面的条款已经将过的):

1>适用于B对象的每一件事,也使用于D对象,因为每个D对象 is-a
B对象。

2>D一定会继承B的non-virtual函数的接口和实现。

所以,如果我们重新定义了一个non-virtual函数:

1>D以public方式继承B,那么当D有一个函数需要重新定义(不变性凌驾特异性),这个函数

应该为virtual的而不是non-virtual的。

2>如果这个函数不需要被重定义(特异性凌驾不变性),那么D久不需要重定义mf,而且它也

不应该尝试这么做。

所以结论是:不要重定义继承而来的non-virtual函数。

*/

}

//#37   绝不重定义继承而来的缺省参数值

{

//  1:看下这两个函数:

class Shape

{

public:

enum ShapeColor{Red, Green, Blue};

virtual void draw(ShapeColor color = Red)
;

};

class Rectangle :
public Shape

{

public:

virtual void draw(ShapeColor color = Green)
const;

};

/*

这么做的结果就是:当客户使用基类指针,子类对象时,调用的内容是子类的,默认参数确实基类的。

这样子做没有意义,所以问题朝着两个方向发展:

1>你想切换默认参数。

2>你想保留默认参数,但是切换调用的内容。如果子类也要使用相同的参数,那么如果默认参数一改变

基类和子类的都必须改变,所以这很糟糕。

实现起来很简单嘛:

1>想切换参数?定义一个non-virtual函数,然后用virtual的函数调用之,

调用时传入不同的参数就ok了。

2>想切换调用内容?定义一个private virtual函数,然后用带着默认参数的non-virtual函数调用之。

*/

}

//#38   通过复合塑膜出has-a或"根据某物实现出"

{

/*

复合有两种情况:

1>has-a

如果你塑造的对象是世界中的某些事物,比如人、物、汽车等等,那么这样的对象属于应用域,

当复合发生在应用域时,表现出has-a的关系。

2>is-implemented-in-terms-of(根据某物实现出)

如果你塑造的对象是为了实现细节上的人工制品,像缓冲区、互斥锁等,这样的对象属于实现域

这样的复合表现出is-implemented-in-terms-of关系。

具体的例子就是,当你想要实现一个set时,你可能需要在类内含一个std::list,因为你不想

自己写list,这样的关系就是is-implemented-in-terms-of,这个set是根据list实现的

这里不用继承的原因是:set不是一个list,list能用的地方set不一定能用。比如list可以插入

重复的值而set不行。

*/

}

//#39   明智而审慎地使用private继承

{

/*

1:private继承意味着is-implemented-in-terms-of。也就是和复合有着异曲同工之妙,然而哪个更好呢?

1>复合可以防止virtual函数被重定义。

2>复合可以解耦两个类,减少编译依赖性。

3>private继承可以实现EBO,也就是使一个“空的”基类真的不占内存。

4>private可以访问到基类protected的内容,复合不行。

2:最后得出的结论:

1>首先考虑复合。

2>真的无法用复合实现,或者真的看重那4byte的字节,那就换成private吧。

*/

}

//#40   明智而审慎地使用多重继承

{

/*

1:多继承常常有很多缺点:

1>比单一继承复杂,可能会造成歧义性,也就是继承的两个基类中有相同名字的函数,那么单单使用

函数名是不知道调用哪一个的。

2>要求基类都来自virtual继承,可是virtual继承是以增加体积,减慢速度,以及复杂的初始化

为成本实现的如果virtual base classes不带任何数据,将是最具使用价值的情况。

(ps:都不带数据了,我不是virtual继承有影响么?就算不是virtual继承也不会有重复东西了啊,

最多就是一个virtual指针)

2:多重继承正当的用途:设计模式中的适配器模式

但是这也是可以转变为复合方式来实现的,所以你需要明智而审慎地决定是否适用多重继承(ps:不建议使用)

*/

}

effective c++ 笔记 (35-40)的更多相关文章

  1. [Effective JavaScript 笔记]第40条:避免继承标准类

    ECMAScript标准库里配备了许多重要的类,如Array,function,以及Date等.扩展这些类生成子类可以方便完成很多工作,但它们的定义具有很多特殊的行为,所以很难写出行为正确的类. Ar ...

  2. leetcode笔记——35.搜索插入位置 - CrowFea

    0.问题描述 给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引.如果目标值不存在于数组中,返回它将会被按顺序插入的位置. 你可以假设数组中无重复元素. 示例 1: 12 输入: [1,3 ...

  3. Effective Java笔记一 创建和销毁对象

    Effective Java笔记一 创建和销毁对象 第1条 考虑用静态工厂方法代替构造器 第2条 遇到多个构造器参数时要考虑用构建器 第3条 用私有构造器或者枚举类型强化Singleton属性 第4条 ...

  4. [Effective JavaScript 笔记] 第4条:原始类型优于封闭对象

    js有5种原始值类型:布尔值.数字.字符串.null和undefined. 用typeof检测一下: typeof true; //"boolean" typeof 2; //&q ...

  5. [Effective JavaScript 笔记] 第5条:避免对混合类型使用==运算符

    “1.0e0”=={valueOf:function(){return true;}} 是值是多少? 这两个完全不同的值使用==运算符是相等的.为什么呢?请看<[Effective JavaSc ...

  6. [Effective JavaScript 笔记]第3章:使用函数--个人总结

    前言 这一章把平时会用到,但不会深究的知识点,分开细化地讲解了.里面很多内容在高3等基础内容里,也有很多讲到.但由于本身书籍的篇幅较大,很容易忽视对应的小知识点.这章里的许多小提示都很有帮助,特别是在 ...

  7. [Effective JavaScript 笔记]第27条:使用闭包而不是字符串来封装代码

    函数是一种将代码作为数据结构存储的便利方式,代码之后可以被执行.这使得富有表现力的高阶函数抽象如map和forEach成为可能.它也是js异步I/O方法的核心.与此同时,也可以将代码表示为字符串的形式 ...

  8. [Effective JavaScript 笔记]第28条:不要信赖函数对象的toString方法

    js函数有一个非凡的特性,即将其源代码重现为字符串的能力. (function(x){ return x+1 }).toString();//"function (x){ return x+ ...

  9. java effective 读书笔记

    java effective 读书笔记 []创建和销毁对象 静态工厂方法 就是“封装了底层 暴露出一个访问接口 ” 门面模式 多参数时 用构建器,就是用个内部类 再让内部类提供构造好的对象 枚举 si ...

  10. Effective STL 笔记 -- Item 6 ~ 7: Container and Object Pointer

    Effective STL 笔记 – Item 6 ~ 7: Container and Object Pointer 中间两次笔记被删掉了,简单补一下: Item 3 中提到如果将对象直接放入容器中 ...

随机推荐

  1. python自学——文件处理(强制刷新)

    # 文件的刷新flash# 为什么要刷新文件呢?# 首先我们就要知道电脑是怎么储存信息的,# 写的代码保存在缓存中当缓存满了之后就会将内容储存到硬盘中. # 那这个跟刷新有什么关系呢?# 系统也会自动 ...

  2. chmod chown llinux文件及目录的权限介绍

    linux 文件或目录的读.写.执行权限说明: chmod :设置文件或目录权限. u:所有者 g:所在组 o:其他组 a:所有人(u.g.o的总和) chmod  -R  文件1/文件2…..    ...

  3. python的学习之路day7-socket网络编程

    python基础部分学习完了,时间也已经过了两个月左右,感觉没学到什么,可能是我学习之后忘记的太多了. 由于没钱买书,要是去培训就更没钱了,所以在网上找了一本书,感觉还不错,讲的比较好,比较详细. P ...

  4. 团队-UML

    UML设计 分工 刘双玉 李佳铭 杜宏庆 肖小强 汪志彬 江郑 符天愉 邓弘立 后台数据库 求购模块 浏览检索商品 即时聊天系统 商品管理 管理员系统 后台商品发布收藏系统 登录注册与个人信息系统 U ...

  5. sublime设置node.js编译

    1. 首先需安装node环境并配置好环境变量,安装教程. 2. 然后在sublime中打开工具(Tools)→编译系统(Build System)→新编译系统(New Build System) 3. ...

  6. java 计算百分数方法

    俗话说好记性不如烂笔头,故记之. DecimalFormat decimalFormat = new DecimalFormat("##.00%"); System.out.pri ...

  7. PHP缓存锁原理及利用

    原文链接:https://blog.csdn.net/tim_phper/article/details/54949404 概述: 项目当中经常要考虑数据高并发的情况,为了避免并发导致出现一些资源重复 ...

  8. 包(package),继承

    1.包(package) 1)为何用包 包用于管理程序中的类,主要解决类同名问题(它的唯一性),也可以看作是现实生活中的目录. 2)作用 —可以解决包的同名问题. —可以更好地管理类,有了包的概念,使 ...

  9. ORACLE 11GR2常用命令

    一.ORACLE的启动和关闭 1.在单机环境下 要想启动或关闭ORACLE系统必须首先切换到ORACLE用户,如下 su - oracle a.启动ORACLE系统 oracle>svrmgrl ...

  10. leetcode 200. Number of Islands 、694 Number of Distinct Islands 、695. Max Area of Island 、130. Surrounded Regions

    两种方式处理已经访问过的节点:一种是用visited存储已经访问过的1:另一种是通过改变原始数值的值,比如将1改成-1,这样小于等于0的都会停止. Number of Islands 用了第一种方式, ...