Part 8: 面向对象(第15章)

// @author:       gr
// @date: 2015-01-09
// @email: forgerui@gmail.com

一、OOP

面向对象程序设计的核心思想是数据抽象、继承和动态绑定。

数据抽象:可以将类的接口与实现分离

继承:可以定义相似的类型并对其相似关系建模

动态绑定:在一定程度上忽略相似类型的区别,使用统一的方式使用它们的对象

二、定义基类或派生类

  1. 基类通常应该定义一个虚析构函数,即使该函数不执行任何实际操作。

  2. virtualoverride关键字只允许在类内使用,类外使用会报错。

  3. 派生类中,C++11标准允许使用override关键字显式地注明它使用某个成员函数覆盖了它继承的虚函数。

  4. C++11中可以使用final关键字防止继承的发生。

     class Last final {
    private:
    int a;
    };
  5. 在覆盖基类的函数时,大意将参数列表写错,没有构成覆盖,将会是两个独立的函数。如果使用override关键字,编译器会报marked override, but does not override的错误,提示有当前的问题。使用override需要保持基类和派生类的函数形式一致,并且基类的函数要声明为virtual

  6. 派生类可以转换到基类,但不存在基类像派生类的转换。从派生类向基类的转换只对指针或引用类型有效。

三、虚函数

  1. 回避虚函数的机制

    强迫执行虚函数的某个特定版本,使用作用域运算符可以实现这一目的:

     double undiscounted = baseP->Quote::net_price(42);  //使用基类Quote中的net_price函数
  2. 多态性

    使用虚函数,通过动态绑定实现,在运行时进行解析。

四、抽象基类

抽象类不能创建对象,只能作为接口,其它类继承它。

含有纯虚函数的类是抽象类。

五、访问控制与继承

  1. 派生类的成员或友元只能通过派生类对象来访问基类的受保护成员。派生类对于一个基类对象中的受保护成员没有任何访问特权。

     class Base{
    protected:
    int prot_mem;
    };
    class Sneaky : public Base{
    friend void clobber(Sneaky&);
    friend void clobber(Base&);
    int j;
    }; //正确:clobber能访问Sneaky对象的private和protected成员
    void clobber(Sneaky& s){s.j = s.prot_mem = 0;}
    //错误:clobber不能访问Base的protected成员
    void clobber(Base& b){b.prot_mem = 0;}

    上面的第二个clobber函数因为不是Base的友元,所以无法访问Base的protected成员。

  2. 派生访问说明符的目的是控制派生类用户(包括派生类的用户、派生类的派生类)对于基类成员的访问权限。

     class Base{
    public:
    void pub_mem();
    protected:
    int prot_mem;
    privated:
    int priv_mem;
    };
    class Pub_Derv : public Base{
    int f() {return prot_mem;}
    char g() {return priv_mem;}
    };
    class Priv_Derv : private Base{
    int f1() const {return prot_mem;}
    };
    Pub_Derv d1;
    Priv_Derv d2;
    d1.pub_mem(); //正确:pub_mem在派生类中是public的
    d2.pub_mem(); //错误:pub_mem在派生类中是private的

    这样,private派生访问说明符表明继承的成员是private,用户无法访问这个成员。使用public继承便可以访问。

    class默认使用的是private继承,struct默认使用的public继承。

  3. 友元与继承

    友元关系不能继承。基类的友元在访问派生类成员时不具有特殊性,同样,派生类的友元也不能随意访问基类的成员。

     class Base{
    friend class Pal; //Pal在访问Base的派生类不具有特殊性
    };
    class Pal{
    int f(Base b){return b.prot_mem;}
    };

六、继承中的类作用域

在编译时进行名字查找

作用域从派生类到基类查找成员。

如果两个成员名字相同,派生类的名字将隐藏基类的名字,但基类中的变量仍然存在。使用作用域运算符可以使用隐藏的名字。

名字查找优先于类型检查

struct Base{
int memfcn();
};
struct Derived : Base{
int memfcn(int);
}
Base b;
Derived d;
b.memfcn(); //正确,调用Base::memfcn()
d.memfcn(42); //正确,调用Derived::memfcn(int)
d.memfcn(); //错误,调用Derived::memfcn(int),参数类型不一致
d.Base::memfcn(); //正确,调用Base:memfcn()

从上面可以看出,查找到相同的名字就停止查找,所以基类的函数被隐藏了。之后再进行参数匹配,发现匹配不对,将会报错。

覆盖重载的函数

派生类可以覆盖重载函数的0个或多个实例。如果派生类希望所有的重载版本对于它来说都是可见的,那么它就要覆盖所有的版本,或者一个都不覆盖

有时,只需覆盖一些而非全部函数时,不都不覆盖每个基类中的每个版本。

解决方案是为重载的成员提供一条using声明语句,使用using可以将子类的所有重载函数放入到当前作用域,这时在添加需要覆盖的函数。

class Base{
public:
//定义三个版本的fcn函数
virtual void fcn();
virtual void fcn(int);
virtual void fcn(double);
};
class Derived : private Base{
public:
//使用using引入Base中的fcn重载函数
using Base::fcn;
//只覆盖int版本
virtual void fcn(int a);
};

七、构造函数与拷贝控制

  1. 基类虚析构函数
  2. 合成拷贝控制与继承

    如果基类的函数是不可访问或者删除的,则派生类的函数也将是被删除的。

八、容器与继承

  1. 容器中存放基类型,将派生对象转换为基对象,派生部分将被“切掉”。

  2. 解决上面的问题,可以在容器中存放(智能)指针而非对象

     vector<shared_ptr<Quote>> basket;
    basket.push_back(make_shared<Quote>("1233", 50));
    basket.push_back(make_shared<Bulk_Quote>("12", 50, 10, .25));
    cout << basket.back()->net_price() << endl;

### 学习《C++ Primer》- 8的更多相关文章

  1. Python学习--04条件控制与循环结构

    Python学习--04条件控制与循环结构 条件控制 在Python程序中,用if语句实现条件控制. 语法格式: if <条件判断1>: <执行1> elif <条件判断 ...

  2. Python学习--01入门

    Python学习--01入门 Python是一种解释型.面向对象.动态数据类型的高级程序设计语言.和PHP一样,它是后端开发语言. 如果有C语言.PHP语言.JAVA语言等其中一种语言的基础,学习Py ...

  3. Python 学习小结

    python 学习小结 python 简明教程 1.python 文件 #!/etc/bin/python #coding=utf-8 2.main()函数 if __name__ == '__mai ...

  4. Python学习路径及练手项目合集

    Python学习路径及练手项目合集 https://zhuanlan.zhihu.com/p/23561159

  5. python学习笔记-python程序运行

    小白初学python,写下自己的一些想法.大神请忽略. 安装python编辑器,并配置环境(见http://www.cnblogs.com/lynn-li/p/5885001.html中 python ...

  6. Python学习记录day6

    title: Python学习记录day6 tags: python author: Chinge Yang date: 2016-12-03 --- Python学习记录day6 @(学习)[pyt ...

  7. Python学习记录day5

    title: Python学习记录day5 tags: python author: Chinge Yang date: 2016-11-26 --- 1.多层装饰器 多层装饰器的原理是,装饰器装饰函 ...

  8. [Python] 学习资料汇总

    Python是一种面向对象的解释性的计算机程序设计语言,也是一种功能强大且完善的通用型语言,已经有十多年的发展历史,成熟且稳定.Python 具有脚本语言中最丰富和强大的类库,足以支持绝大多数日常应用 ...

  9. Python学习之路【目录】

    本系列博文包含 Python基础.前端开发.Web框架.缓存以及队列等,希望可以给正在学习编程的童鞋提供一点帮助!!! 目录: Python学习[第一篇]python简介 Python学习[第二篇]p ...

  10. python学习笔记系列----(八)python常用的标准库

    终于学到了python手册的最后一部分:常用标准库.这部分内容主要就是介绍了一些基础的常用的基础库,可以大概了解下,在以后真正使用的时候也能想起来再拿出来用. 8.1 操作系统接口模块:OS OS模块 ...

随机推荐

  1. 【C语言】-条件语句-switch语句

    switch语句: 用于直接处理不同情况下的多路问题. switch语句又可称为开关语句,其执行流程和多分支if语句类似. switch (表达式) { case 常量表达式1:语句组1;break; ...

  2. URL编码原理解释

    当你在浏览器中输入一个URL时,浏览器会将你输入到地址栏的非数字字母转化为URI编码. 那么,它是按照什么样的规则来转换的呢 是这样的,URI编码就是一个字符的ASCII码,它的ACSII码的十六进制 ...

  3. JQuery事件处理的注意事项

    1.jQuery 名称冲突 jQuery 使用 $ 符号作为 jQuery 的简介方式. 某些其他 JavaScript 库中的函数(比如 Prototype)同样使用 $ 符号. jQuery 使用 ...

  4. mysql之存储引擎

    1.存储引擎概念 打比方说:一部电影有mp4,wmv,avi,flv...等格式.同样的一部电影在硬盘上有不同的存储格式,所占的空间与清晰程度也各不一样. 那么我们表里的数据存储在硬盘上,是如何存储的 ...

  5. XMPP——Smack[6]离线消息和离线文件的实现

    终篇,三天所学所用,也就这些,如果需要大家要自己去查资料研究研究,功能其实可以很强大的 可惜界面做得不好,一大短处,从大一迄今没整好,主要是个人审美不行,哎 毕业季呀毕业季,明天摆摊卖书,再半月就可能 ...

  6. 获取WMI硬件清单

    WMI服务能够报告详细的硬件信息.通常,每个硬件都来自它们自己的WMI代理类.但是要找出这些硬件类的名字是不容易. 所有硬件类都在同一个WMI根下面,你可以在根类查询所有的硬件: Get-WmiObj ...

  7. 9个使用前必须再三小心的Linux命令

      Linux shell/terminal 命令非常强大,即使一个简单的命令就可能导致文件夹.文件或者路径文件夹等被删除.在一些情况下,Linux 甚至不会询问你而直接执行命令,导致你丢失各种数据信 ...

  8. 初探 MySQL 的 Binlog

    https://xcoder.in/2015/08/10/mysql-binlog-try/

  9. 《RESTful Web Services》第一章 使用统一接口

    序言 HTTP是一种应用层协议.SOAP和一些Ajax Web框架都将HTTP作为一种传输信息的协议,难以充分利用HTTP层的基础设施. 1.2 如何保持交互的可见性     可见性是HTTP的一个核 ...

  10. Mysql性能调优

    在MYSQL命令行客户端添加 \G 语句终止符可以让返回的结果集垂直显示. 一.查找运行缓慢的 SQL语句 :show full processlist ; 二.生成一个查询执行计划(Query Ex ...