c++基础(七)——面向对象程序设计
面向对象程序设计(Object-oriented programming)的核心思想是数据抽象,继承,和动态绑定。
1. 继承
在C++语言中,基类将类型相关的函数与派生类不做改变直接继承的函数区分对待。对于某些函数,基类希望它的派生类各自定义适合自身的版本,此时基类就将这些函数声明成虚函数(virtual function)。
class Quote {
public:
std::string isbn() const;
virtual double net_price(std::size_t n) const;
};
派生类必须通过使用派生列表(class derivation list)明确指出它是从哪个(哪些)基类继承而来的:
如下:
class Bulk_quote : public Quote {
public:
double net_price(std::size_t) const override;
};
派生类必须在其内部对所有重新定义的虚函数进行声明。派生类可以在这样的函数之前加上virtual关键字,但是并不是非得这么做。并且,C++11新标准允许派生类显式地注明它将使用哪个成员函数改写基类的虚函数,具体措施是在改函数的形参列表之后增加一个override关键字。
1.1 访问控制和继承
#include<iostream>
class A {
public:
int p = ;
virtual void test();
private: int p2 = ;
protected: int p3 = ;
}; void A::test()
{
std::cout << this->p << this->p2 << this->p3 << std::endl;
} class B:public A {
public:
int b = ;
void test() {
std::cout << this->b << this->b2 << this->b3 << std::endl;
} void test2() {
std::cout << this->p3 << std::endl; // 派生类可以访问基类的protect和public
} friend void test3() {
std::cout << this-> << std::endl;
}
private: int b2 = ;
protected: int b3 = ;
}; int main()
{
A a;
std::cout << a.p << std::endl;// 只能访问自己的public
a.test(); B b;
std::cout << b.b << b.p << std::endl;// 派生类 只能访问自己的puiblic和基类的public
b.test(); }
1.2 定义基类和派生类
1.定义基类。
Note:基类通常都应该定义一个虚析构函数,即使该函数不执行任何实际操作也是如此。
class Quote {
public:
Quote() = default;
Quote(const std::string &book, double sales_price):
bookNo(book), price(sales_price){}
std::string isbn()const {return bookNo;}
//返回给定数量的书籍的销售总额
//派生类负责改写并使用不同的折扣计算算法
virtual double net_price(std::size_t n) const
{return n*price;}
virtual ~Quote() = default; //对析构函数进行动态绑定
private:
std::string bookNo;
protected:
double price = 0.0; //代表普通状态下不打折的价格
};
- 基类必须将它的两种成员函数区分开来:一种是基类希望其派生类进行覆盖的函数,基类通常将其定义为虚函数;另一种是基类希望派生类直接继承而不要改变的函数。
- 任何构造函数之外的非静态函数都可以是虚函数,关键字virtual只能出现在类内部的声明语句之前而不能用于类外部的函数定义。
- 如果基类把一个函数声明为虚函数,则该函数在派生类中隐式地也是虚函数。
- 成员函数如果没被声明为虚函数,则其解析过程发生在编译时而非运行时。
- 派生类能访问公有成员,而不能访问私有成员。
- 不过在某些时候,基类中还有一种成员,基类希望它的派生类有权访问该成员,同时禁止其他用户访问。我们用受保护的(protected)访问运算符说明这样的成员。
2. 定义派生类
class Bulk_quote : public Quote {
public:
Bulk_quote() = default;
Bulk_quote(const std::string&, double, std::size_t, double);
//覆盖基类的函数版本以实现基于大量购买的折扣政策
double net_price(std::size_t) const override;
private:
std::size_t min_qty = ; //使用折扣政策的最低购买量
double discount = 0.0; //以小数表示的折扣额
};
- 派生类经常(但不总是)覆盖它继承的虚函数。如果派生类没有覆盖其基类中的某个虚函数,则该虚函数的行为类似于其他的普通成员,派生类会直接继承其在基类中的版本。
- C++新标准允许派生类显式地注明它使用某个成员函数覆盖了它继承的虚函数。具体做法是在函数后面加上关键字override。
- 在派生类对象中含有与其基类对应的组成部分,所以我们能把派生类的对象当成基类对象来使用,而且我们也能将基类的指针或引用绑定到派生类对象中的基类部分上。如下:
Quote item; //基类对象
Bulk_quote bulk; //派生类对象
Quote *p = &item; //p指向Quote对象
p = &bulk; //p指向bulk的Quote部分
Quote &r = bulk; //r绑定到bulk的Quote部分
3. 派生类构造函数
派生类可以继承基类的成员,但是不能直接初始化这些成员(每个类控制它自己的成员初始化过程)。如果没有在子类中对父类进行初始化,则父类必须有默认构造函数。
Bulk_quote(const std::string &book, double p, std::size_t qty,
double disc):Quote(book, p), min_qty(qyt), discount(disc) { }
4.派生类使用基类的成员
派生类可以访问基类的公有成员,和受保护成员
5.继承与静态成员
如果基类中有静态变量,则不论派生出多少类,对每个静态成员来说都只存在唯一实例。如果基类中静态成员是private,则派生类无权访问,假设派生类可以访问,则我们既能通过基类也能通过派生类使用它。
6.防继承
C++11新标准提供了一种防止继承发生的方法,即在类名后跟一个关键字final
class NoDerived final { /* */}; //NoDerived不能作为基类
7.不存在从基类向派生类的隐式类型转换
之所以存在派生类向基类的类型转换是因为每个派生类都包含了基类的一部分,而基类引用或者指针可以绑定到该基类部分上。但是因为一个基类对象可能是派生类对象的一部分,也可能不是,所以不存在从基类向派生类的自动类型转换。
Quote base;
Bulk_quote* blukP = &base; //不合法
Bulk_quote& blukRef = base; //不合法
下面这种情况也是不允许的
Bulk_quote bulk;
Quote* itemP = &bulk; //合法,基类绑定派生类
Bulk_quote* blukRef = itemP; //不合法
如果基类中含有一个或多个虚函数,我么可以使用 dynamic_cast 请求类型转换。
8.对象之间不存在类型转换
派生类向基类的自动类型转换只对指针或引用类型有效,在派生类类型和基类类型之间不存在这样的转换。很多时候,我们确实希望将派生类对象转换成基类类型,但是这种转换往往与我们所期望的不一样。
请注意,当我们初始化或赋值一个类类型的对象时,实际上实在调用某个函数。当执行初始化时,我们调用构造函数,而当执行赋值操作时,我们调用赋值运算符。
因为这些成员接受引用作为参数,多以派生类向基类的转换允许我们给基类的拷贝和移动操作传递一个派生类的对象。这些操作不是虚函数,当我们给基类的构造函数传递一个派生类对象时,实际运行的构造函数是基类中定义的那个,显然,该运算符只能处理基类自己的成员。
Bulk_quote bulk; // 派生类对象
Quote item(bulk);// 使用Quote::Quote(const Quote&)构造函数
item = bulk; // 调用Quote::opertator=(const Quote&)。同时忽略掉派生类中的部分成员
2. 动态绑定
在C++语言中,当我们使用基类的引用(或指针)调用一个虚函数时将发生动态绑定。明晰派生类调用到底调用谁的print方法。
比如:
#include<iostream>
class A {
public:
A() = default;
virtual void print() {
std::cout << "a" << std::endl;
}
virtual ~A() {
std::cout << "destroy A" << std::endl;
};
}; class B :public A {
public:
B() = default;
void print() {
std::cout << "b" << std::endl;
}
~B() {
std::cout << "destroy B" << std::endl;
} }; int main()
{
A a;
B b;
a.print();
b.print(); /// 动态绑定.如果基类中 print 方法不是虚函数,则以下结果均为a
A *a2 = &a;
A *b2 = &b;
a2->print();// a
b2->print();// b A &a3 = a;
A &b3 = b;
a3.print();// a
b3.print();// b /// 强行调用基类
//b.A::print(); // a
//b2->A::print(); // a
//b3.A::print(); // a }
3. 虚函数
如第二点所述,当我们使用基类的引用或指针调用一个虚函数时会执行动态绑定。因为我们直到运行时,才能知道到底调用了哪个版本的虚函数(上面的例子能看出,但是编译器看不出),所以所有虚函数都必须定义。需要注意的,动态绑定必须通过指针,引用,调用虚函数才会发生。如果使用类类型【如:Quote base; Bulk_Quote derived; base=derived; base.net_price(21)。其中net_price是基类虚函数,派生类重写了。这个情况下编译时,会被解析成基类的net_price() 方法。】非虚函数在编译时进行绑定。
一个派生类的函数如果覆盖了某个继承来的虚函数,则它的形参类型必须与被覆盖的基类函数完全一致。
3.1 final和override说明符
派生类如果定义了一个和基类中名字相同但是形参列表不同,这也是合法的。编译器会认为这是两个函数与基类中原有的函数是相互独立的。但是这可能不是我们想要的,我们想要覆盖基类方法,但是编译器不报错。这是c++11 中新出了 overrride 关键字,这可以让编译器明白我们的用意,并为我们发现错误(形参是否写错了等)。
如果我们用 override 关键字标记了某个函数,但是该函数没有覆盖已存在的虚函数,此时编译器会报错。
如果使用 final 关键词标记函数,则不允许后序其他类覆盖该方法。
struct C {
virtual void f1() const;
}; struct D:C
{
void f1() const final; // final 修饰虚函数
};
void D::f1() const { }
struct E:D
{
void f1() const ; // 错误
};
4.抽象基类
4.1 纯虚函数
为什么要有纯虚函数?因为在许多情况下,在基类中不能对虚函数给出有意义的实现(比如动物基类中“叫”的方法),而把它声明为纯虚函数,它的实现完全留给派生类去做。凡是含有纯虚函数的类叫做抽象类,这种类不能声明对象,只能作为基类为派生类服务。
virtual void f3()=;
派生类构造函数只初始化它的直接基类。
5. 访问控制与继承
5.1 受保护的成员
类中的私有变量只能由该类的成员方法能访问。一个类使用protected关键字来声明哪些希望与派生类分享但是不想被其他公共访问使用的成员,protected可以看成是private 和 public的中和产物:
- 和私有成员类似,受保护的成员对于类的用户来说不可访问
- 和公有成员类似,受保护的成员对于派生类的成员和友元来说是可访问的
- 派生类的成员或友元只能通过派生类对象来访问基类的受保护成员。派生类对于一个基类对象中的受保护成员没有任何访问权
class Base {
protected:
int prot_mem; //protected成员
};
class Sneaky : public Base {
friend void clobber(Sneaky&); //能访问Sneaky::prot_mem
friend void clobber(Base&); //不能访问Base::prot_mem
int j; //j默认是private
};
//正确:clobber能访问Sneaky对象的private和protected成员
void clobber(Sneaky &s) { s.j = s.prot_mem = ; }
//错误:clobber不能访问Base的protected成员
void clobber(Base &b) { b.prot_mem = ; }
- 在类的外部(比如main函数中),类的pirvate成员不管是对该类的对象还是该类派生类的对象,都是无访问权限的
5.2 公有,私有和受保护继承
#include<iostream>
using namespace std; class base {
public:
int pub;
void pub_fun_base() {
cout << "base pub_fun" << endl;
}
private:
int pri;
void pri_fun_base() {
cout << "base pri_fun" << endl;
}
protected:
int pro;
void pro_fun_base() {
cout << "base pro_fun" << endl;
}
}; struct pub_derived : public base
{
void f() {
cout << pub << pro << endl; // 不能访问父类的私有变量 pri
pub_fun_base();
pro_fun_base();
} }; struct pri_derived : private base // 派生类说明符private同public看似访问权限一样,那该说明符何用?(在main函数中解释)
{
void f() {
cout << pub << pro << endl;
pub_fun_base();
pro_fun_base();
}
}; struct pro_derived : protected base
{
void f() {
cout << pub << pro << endl;
pub_fun_base();
pro_fun_base();
}
}; int main()
{
base base1;
base1.pub;
base1.pub_fun_base(); // 说明基类对象能直接访问public成员,protect/private不可。在成员方法中均可 pub_derived pub_d;
pri_derived pri_d;
pro_derived pro_d;
pub_d.f();
pri_d.f();
pro_d.f(); pub_d.pub_fun_base(); // 派生类说明符作用在此,控制基类中继承过来的公有成员是保持public还是改成private/public
//pri_d.pub_fun_base(); //非法 // 基类中的public成员通过private说明符继承,则该方法不能由派生类对象直接访问了,相当于变成了派生类中的private成员
//pro_d.pub_fun_base(); //非法
return ;
}
5.3 虚析构函数
#include<iostream>
using namespace std; struct base {
base() { cout << "base create" << endl; }
~base(){ cout << "base destroy" << endl; }
}; struct derived_base : public base {
derived_base() { cout << "derived_base create" << endl; }
~derived_base() { cout << "derived_base destroy" << endl; }
}; int main()
{ derived_base* db = new derived_base();
delete db; return ;
}
下面更改下代码如下:
#include<iostream>
using namespace std; struct base {
base() { cout << "base create" << endl; }
virtual ~base(){ cout << "base destroy" << endl; }
}; struct derived_base : public base {
derived_base() { cout << "derived_base create" << endl; }
~derived_base() { cout << "derived_base destroy" << endl; }
}; int main()
{
base* b = new derived_base(); // 使用动态绑定
delete b; return ;
}
如果没有使用动态绑定,就是第一个例子。如果使用了动态绑定,但是如果没有第二个例子第六行在基类中定义虚析构函数,则没有截图中红色箭头销毁派生类的操作。为什么呢?因为我们知道第一个例子是静态绑定,第二个例子是动态绑定,动态绑定在运行时才能确定执行基类or派生类方法,动态绑定的条件时:①使用指针或引用②使用虚函数。所以如果没有在基类中定义虚函数,则析构函数没有采用动态绑定,而是使用静态绑定,则只会执行基类的析构,而不会执行派生类的虚构函数。
c++基础(七)——面向对象程序设计的更多相关文章
- python基础16 ----面向对象程序设计二
一.继承与派生 1.继承的定义:继承是一种创建新类的方式,即在类中提取共同的部分创建出一个类,这样的类称为父类,也可称为基类和超类,新建的类称为派生类或子类. 2.单继承:就相当于子类继承了一个父类. ...
- 王之泰201771010131《面向对象程序设计(java)》第七周学习总结
王之泰201771010131<面向对象程序设计(java)>第七周学习总结 第一部分:理论知识学习部分 第五章 第五章内容深度学习: 继承:如果两个类存在继承关系,则子类会自动继承父类的 ...
- 201771010134杨其菊《面向对象程序设计java》第七周学习总结
第七周学习总结 第一部分:理论知识 1.继承是面向对象程序设计(Object Oriented Programming-OOP)中软件重用的关键技术.继承机制使用已经定义的类作为基础建立新的类定义,新 ...
- 201871010101-陈来弟《面向对象程序设计(java)》第七周学习总结
201871010101-陈来弟<面向对象程序设计(java)>第七周学习总结 项目 内容 <面向对象程序设计(java)> https://www.cnblogs.com/n ...
- 201871010104-陈园园 《面向对象程序设计(java)》第七周学习总结
201871010104-陈园园 <面向对象程序设计(java)>第七周学习总结 项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ ...
- 201871010105-曹玉中《面向对象程序设计(java)》第七周学习总结
201871010105-曹玉中<面向对象程序设计(java)>第七周学习总结 项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这 ...
- 201871010107-公海瑜《面向对象程序设计(java)》第七周学习总结
201871010107-公海瑜<面向对象程序设计(java)>第七周学习总结 项目 内容 这个作业属于哪个课程 ...
- 唐敬博-201871010118 《面向对象程序设计(java)》第七周学习总结
在博客园撰写博客(随笔),总结7周实验内容,作业格式要求如下: 博文名称:学号-姓名<面向对象程序设计(java)>第七周学习总结(1分) 博文正文开头格式:(2分) 项目 内容 这个作业 ...
- 张兴盼-201871010131《面向对象程序设计(Java)》第七周学习总结
张兴盼-201871010131<面向对象程序设计(Java)>第七周学习总结 项目 内容 这个作业属于哪个课程 http://www.cnblogs.com/nwnu-daizh/ 这个 ...
随机推荐
- Boosting and AdaBoost
Boosting是一种从一些弱分类器中创建一个强分类器的集成技术(提升算法). 它先由训练数据构建一个模型,然后创建第二个模型来尝试纠正第一个模型的错误.不断添加模型,直到训练集完美预测或已经添加到数 ...
- C# 监测每个方法的执行次数和占用时间(测试4)
今天也要做这个功能,就百度一下,结果搜索到了自己的文章.一开始还没注意,当看到里面的一个注释的方法时,一开始还以为自己复制错了代码,结果仔细一看网页的文章,我去,原来是自己写的,写的确实不咋地. 把自 ...
- 洛谷 P3385 【模板】负环 题解
P3385 [模板]负环 题目描述 暴力枚举/SPFA/Bellman-ford/奇怪的贪心/超神搜索 寻找一个从顶点1所能到达的负环,负环定义为:一个边权之和为负的环. 输入格式 第一行一个正整数T ...
- 【loj2567】【APIO2016】划艇
题目 \(N\)个位置,每个位置要么不选,要么选\([ a_i, b_i ]\)中的一个数: 问最后的单调上升序列(mod 1e9+7)有多少种: \(1 \le N \le 500\) 题解 orz ...
- sublime text 3插件改造之AutoFileName去掉.vue文件中img标签后面的width和height,完全去掉!!
在.vue文件中img标签使用autofilename提示引入文件时,会在文件后面插入宽度高度,如下图: 文件后面会自动插入height和width,其实这两玩意儿在大多数时候并没卵用,然后就开始了百 ...
- 解决Git - git push origin master 报错
关注我,每天都有优质技术文章推送,工作,学习累了的时候放松一下自己. 欢迎大家关注我的微信公众号:「醉翁猫咪」 原因:github仓库中没有README.md文件 解决如下: 重新输入git push ...
- mysql rtrim() 函数
mysql> select rtrim(" cdcdcd "); +--------------------+ | rtrim(" cdcdcd ") | ...
- python设计模式---绪论
1.程序只是一个工具,只知道使用工具就有价值的时代正在过去:现在对工作质量.开发速度及完美程度都很重要了.当前主要的问题是对工具的充分利用,在生活的方方面面,简单任务之所以简单是由于这些任务不需要特殊 ...
- tomcat监控工具-probe
概述 今天给大家介绍一款开袋即食的性能监控工具,居家性能测试必备! tomcat监控工具:probe tomcat probe是一个开源的监控tomcat运行状态工具,可以实时查看项目运行的情况,监控 ...
- 把pdf的内容转化为txt文件
import org.apache.pdfbox.pdmodel.PDDocument; import org.apache.pdfbox.util.PDFTextStripper; import j ...