32.确保你的public继承了模is-a关系

public继承意味着is-a关系(里氏替换原则),一切适用于基类也适用于派生类。

矩形继承正方形问题:

  • 可实施与矩形的操作无法实施与正方形
  • 在编程领域。正方形是一种矩形是错误的
  • 在现实领域,正方形是一种矩形是正确的

33.避免遮盖继承而来的名称

class Base {
private:
int x;
public:
virtual void mf1() = 0;
virtual void mf1(int);
virtual void mf2();
void mf3();
void mf3(double);
};
class Derived:public Base {
public:
virtual void mf1();
void mf3();
void mf4();
}; Derived d;
int x;
d.mf1(); // 调用Derived::mf1
d.mf1(x); // 错误,由于Derived::mf1遮掩了Base::mf1
d.mf2(); // 调用Base::mf2
d.mf3(); // 调用Derived::mf3
d.mf3(x); // 错误,由于Derived::mf3遮掩了Base::mf3

继承类函数会遮掩基类的同名函数。即使參数不同。

目的是防止继承类从基类继承重载函数。

解决方法例如以下:

class Derived:public Base {
public:
using Base::mf1;// 让Base class内mf1的全部东西在继承类作用域中都可见
using Base::mf3;
virtual void mf1();
void mf3();
void mf4();
};

为了让遮掩重见天日。可使用using声明式或者转交函数。

34.区分接口继承和实现继承

  1. 接口继承
  2. 接口和实现继承,同一时候覆写
  3. 接口和实现继承。不覆写
class Shape {
public:
virtual void draw() const = 0;// 接口继承
virtual void error(const string& msg);// 接口和实现继承。同一时候覆写
int object() const;// 接口和实现继承,不覆写
};

同意impure virtual函数同一时候指定函数声明和函数缺省行为。可能造成危急。

类A提供纯虚函数及实现fly()。期望继承类B使用fly实现,继承类C不使用fly实现。C类的实现者可能不清楚这一约定,造成类C也使用fly的实现。

解决方法:

1. 将fly改为纯虚函数。类A中实现protected defaultfly。

类B实现fly中调用defaultfly。类C实现fly。

2. 将fly改为纯虚函数,类A中提供fly的定义,类B实现的fly中调用A::fly(),类C实现fly

结论:

  • 声明为纯虚函数期望继承类仅仅继承接口。
  • 声明为虚函数目的是让继承类继承该函数的接口和缺省默认实现
  • 声明为非虚函数目的是让继承类继承函数的接口和一份强制的实现

35.考虑virtual函数以外的其它选择

  • NVI Non-Virtual Interface用来替代public virtual的方案
  • 由Function Pointers实现strategy模式
  • 古典策略模式

    遗留问题:

    tr1::function使用

36.绝不又一次定义继承而来的non-virtual函数

non-virtual代表不变性凌驾与特异性。

目的是令继承类继承函数的接口及一份强制实现

使用基类指针会调用到基类的版本号

37.绝不重定义继承而来的缺省參数

非虚函数中不同意在继承类中重定义

在虚函数中更改继承类的缺省參数不会起作用

虚函数中更改缺省參数不起作用的原因,出于效率考虑,在运行期确定參数比编译器决定慢且更为复杂

虚函数中不要提供缺省參数,虚函数改动缺省參数后,继承类要跟着改动

使用NVI(non-virtual infterace方法。non-virtual方法中提供缺省參数,no-virtaul方法调用virtual方法)

class Shape {
public:
enum ShapeColor {Red,Green,Blue};
void draw(ShapeColor color = Red) const {
doDraw(color);
}
private:
virtual void doDraw(ShapeColor color) const = 0;
};
class Rectangel:public Shape {
private:
virtual void doDraw(ShapeColor color) const;
};

38.通过复合塑模出has-a或“依据某物实现出”

复合的两层含义

  • has-a(应用域对象,比如汽车,视频画面)
  • is-implemented-in-terms-of(实现域,比如缓存器。相互排斥器)

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

  • 假设是private继承,须要基类对象的场合。传入继承类错误,编译器不会自己主动将继承类对象转换为一个基类对象。
  • 由private继承而来的全部成员,在继承类中都会变为private
  • private继承意味着implemented-in-terms-of(依据某物实现)
  • 继承类和基类没有不论什么关系
  • private继承意味着实现被继承,接口部分被略去。private继承在设计层面没有意义
  • 和复合之间的取舍
    • 尽可能使用复合
    • 须要改动private继承基类中的虚函数时使用私有继承
    • 替代私有继承须要更改虚函数的还有一种方法: 使用复合,复合类中定义一个public继承的嵌套类,复合类使用该嵌套了的对象。
class Widget {
private:
class WidgetTimer:public Timer {
public:
virtual void OnTick() const;
};
WidgetTimer timer;
};

以上方式也能够用来阻止继承类又一次定义virtual函数

  • 使用private的还有一个场合:EBO empty base optimization空白基类最优化(STL中使用 unary_function,binary_function)

版权声明:本文博客原创文章,博客,未经同意,不得转载。

Effective C++ -- 继承和面向对象设计的更多相关文章

  1. Effective C++ —— 继承与面向对象设计(六)

    条款32 : 确定你的public继承塑模出is-a关系 以C++进行面向对象编程,最重要的一个规则是:public inheritance(公开继承)意味“is-a”(是一种)的关系.请务必牢记.当 ...

  2. Effective C++ ——继承与面向对象设计

    条款32:确定你的public继承塑模出is-a关系 以public继承的类,其父类的所有的性质都应该使用与子类,任何需要父类的地方都应该能用子类来代替,任何子类类型的对象也同时是父类的: class ...

  3. EffectiveC++ 第6章 继承与面向对象设计

    我根据自己的理解,对原文的精华部分进行了提炼,并在一些难以理解的地方加上了自己的"可能比较准确"的「翻译」. Chapter 6 继承与面向对象设计 Inheritance and ...

  4. Effective C++(20) 继承与面向对象设计

    本文主要参考<Effective C++ 3rd>中的第六章部分章节的内容. 关注的问题集中在继承.派生.virtual函数等.如: virtual? non-virtual? pure ...

  5. Effective C++: 06继承与面向对象设计

    32:确定你的public继承塑模出is-a关系 以C++进行面向对象编程,最重要的一个规则是:public继承表示的是"is-a"(是一种)的关系. 如果令class D以pub ...

  6. Effective C++ 6.继承与面向对象设计

    //条款32:确定你的public继承塑模出is-a关系 // 1.public继承意味着is-a的关系,适用在基类上的方法都能用于派生类上. //条款33:避免遮掩继承而来的名称 // 1.在pub ...

  7. Effective C++笔记:继承与面向对象设计

    关于OOP 博客地址:http://www.cnblogs.com/ronny 转载请注明出处! 1,继承可以是单一继承或多重继承,每一个继承连接可以是public.protected或private ...

  8. Effective C++笔记(六):继承与面向对象设计

    参考:http://www.cnblogs.com/ronny/p/3756494.html 条款32:确定你的public继承塑模出is-a关系 “public继承”意味着is-a.适用于base ...

  9. 【Effective C++】继承与面向对象设计

    关于OOP 1,继承可以是单一继承或多重继承,每一个继承连接可以是public.protected或private,也可以是virtual或non-virtual. 2,成员函数的各个选项:virtu ...

随机推荐

  1. 开放源代码的微微信.NET 0.8 版公布了

    微微信.NET 0.8 版公布了     A.源代码应用范围:         未认证的和经过认证的微信订阅号.微信服务号均可使用,本源代码的每个模块都提供全然的 ASP.NET C#源代码,绝对不含 ...

  2. ubuntu下使用自带的openJDK查看java源码

    如题 Ubuntu自带的OpenJDK仅仅有jre环境,不提供源代码,所以我们还是须要去下载. JDK6:http://download.java.net/openjdk/jdk6/ JDK7:htt ...

  3. GString及IntelliJIdea中调试Groovy的操作步骤

    今天是学习Groovy的第一天,首先我觉得学习任何一种语言都要先弄清楚这种语言的特性,因为只有了解了特性之后学习才能达到好的效果,那么groovy的特点是什么的.我觉得groovy是一种动态语言,动态 ...

  4. NET通用平台

    NET通用平台.通用权限.易扩展.多语言.多平台架构框架 先拿出我半前年前平台的设计初稿,经过半年的努力我已经完成了该设计稿的所有功能.并且理念已经远远超出该设计稿. 下面是一些博友对我贴子的评价: ...

  5. unity3d由于Camera.main.transform报空引用错误的解决方案

    今天在导入character包后,引用了内置的第三人称素材,但是在启动的时候程序报空引用的错误: 引起错误的位置在: 错误原因是因为没有将摄像机的tag设置为maincamera,改为maincame ...

  6. 树上第k小,可持久化线段树+倍增lca

    给定一颗树,树的每个结点都有权值, 有q个询问,每个询问是 u v k ,表示u到v路径上第k小的权值是多少. 每个结点所表示的线段树,是父亲结点的线段树添加该结点的权值之后形成的新的线段树 c[ro ...

  7. HDU 1171 Big Event in HDU (多重背包)

    Big Event in HDU Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others ...

  8. java大全经典的书面采访

    果学网  -专注IT在线www.prismcollege.com 1.面向对象的特征有哪些方面 1.抽象: 抽象就是忽略一个主题中与当前目标无关的那些方面.以便更充分地注意与当前目标有关的方面.抽象并 ...

  9. 递归遍历XML所有节点

    package xml; import org.dom4j.Document; import org.dom4j.DocumentHelper; import org.dom4j.DocumentEx ...

  10. Spring MVC helloWorld中遇到的问题及解决办法

    1.java.io.FileNotFoundException: Could not open ServletContext resource不能加载ServletContext的用法是配置到web. ...