23.C++- 继承的多种方式、显示调用父类构造函数、父子之间的同名函数、virtual虚函数
上章链接: 22.C++- 继承与组合,protected访问级别
继承方式
继承方式位于定义子类的”:”后面,比如:
class Line : public Object //继承方式是public
{
};
继承方式默认为private
在C++中,继承方式共有3种:
public继承
-指父类的成员(变量和函数)访问级别,在子类中保持不变
private继承
-指父类的成员,在子类中变为private私有成员.
-也就是说子类无法访问父类的所有成员
protected继承
-指父类的public成员 ,在子类中变为protected保护成员,其它成员级别保持不变
如下图所示:
注意: protected继承只针对子类有效
比如当父类是protected继承时,则子类的子类就无法访问父类的所有成员
一般而言,C++项目只用到public继承
显示调用父类构造函数
- 当我们创建子类对象时,编译器会默认调用父类无参构造函数
- 若有子类对象,也会默认调用子类对象的无参构造函数。
比如以下代码:
class StrA
{
public:
StrA()
{
cout<<"StrA()"<<endl;
}
StrA(string s)
{
cout<<"StrA(string s):"<<s<<endl;
}
}; class StrB : public StrA
{
public:
StrB(string s)
{
cout<<"StrB(int i):"<<s<<endl;
}
}; int main()
{
StrB b("");
return ;
}
编译运行:
StrA() //父类无参构造函数
StrB(int i):
也可以通过子类构造函数的初始化列表来显示调用
接下来,修改上面子类的StrB(string s)函数,通过初始化列表调用StrA(string s)父类构造函数
改为:
StrB(string s): StrA(s)
{
cout<<"StrB(int i):"<<s<<endl;
}
运行打印:
StrA(string s):
StrB(int i):
父子间的同名成员和同名函数
- 子类可以定义父类中的同名成员和同名函数
- 子类中的成员变量和函数将会隐藏父类的同名成员变量和函数
- 父类中的同名成员变量和函数依然存在子类中
- 通过作用域分辨符(::)才可以访问父类中的同名成员变量和函数
比如:
class Parent{ public:
int mval;
Parent()
{
mval=;
}
void add(int i)
{
mval+=i;
}
}; class Child : public Parent
{
public:
int mval;
Child()
{
mval=;
} void add(int i,int j)
{
mval+=i+j;
}
};
在main()函数执行:
Child c; //c. add(10); //该行会报错,由于子类有add函数,所以编译器会默认在子类里寻找add(int i); c.Parent::add(); //该行正确,执行父类的成员函数 c.add(,); cout<<"Child.mval="<<c.mval<<endl; cout<<"Parent.mval="<<c.Parent::mval<<endl;
打印:
Child.mval=
Parent.mval=
从打印结果看到,父类和子类之间的作用域是不同的, 所以执行父类的同名成员变量和函数需要作用域分辨符(::)才行
父子间的兼容
以上示例的Parent父类Child子类为例
- 子类对象可以直接赋值给父类对象使用,比如: Parent p; Child c; p=c;
- 子类对象可以初始化父类对象,比如: Parent p1(c);
- 父类引用可以直接引用子类对象,比如: Parent& p2 =c; //p2是c对象的别名
- 父类指针可以直接指向子类对象,比如: Parent* p3=&c;
其实是编译器是将子类对象退化为了父类对象, 从而能通过子类来赋值初始化父类
所以上述的父类对象(包括指针/引用)也只能访问父类中定义的成员.
如果父类对象想访问子类的成员,只能通过强制转换,将父类对象转为子类类型
示例1,通过C方式转换:
Child c;
Parent* p3=&c;
Child *c2 = (Child*)p3;
示例2,通过static_cast转换:
Child c;
Parent* p3=&c;
Child *c2 = (static_cast*)<Child*>(p3);
虚函数
实现多态性,通过指向子类的父类指针或引用,可以访问子类中同名覆盖成员函数
首先参考下面,没有虚函数的示例:
class Parent
{
int i;
public:
void example()
{
cout<<"class Parent"<<endl;
} }; class Child : public Parent
{
int j;
public:
void example()
{
cout<<"class Child"<<endl;
}
}; void print(Parent* p)
{
p->example();
}
int main()
{
Parent t;
Child c; print(&t);
print(&c); cout<<"SIZEOF Parent:"<<sizeof(t)<<endl;
cout<<"SIZEOF Child:"<<sizeof(c)<<endl;
}
运行打印:
class Parent
class Parent
SIZEOF Parent:
SIZEOF Child:
从结果看出,即使example函数的指针p指向了Child c,也只能调用父类的example(),无法实现多态性.
所以C++引入了虚函数概念,根据指针指向的对象类型,来执行不同类的同名覆盖成员函数,实现不同的形态
定义: 在父类成员函数的返回值前面,通过virtual关键字声明,这样便能访问子类中的同名成员函数了
接下来将上个示例的父类成员函数example()改写为虚函数:
virtual void print() //将父类的成员函数定为虚函数
{
cout<<"class Parent"<<endl;
}
运行打印:
class Parent
class Child
SIZEOF Parent:
SIZEOF Child:
可以发现,父类和子类的长度都增加了4字节,这4个字节就是用来指向“虚函数表”的指针,编译器便会更据这个指针来执行不同类的虚函数,实现多态性.
虚析构函数
-在使用基类指针指向派生类对象时用到
-通过基类析构函数可以删除派生类对象
示例
#include <iostream> using namespace std; class Base
{
public:
Base()
{
cout << "Base()" << endl;
} virtual ~Base()
{
cout << "~Base()" << endl;
}
}; class Derived : public Base
{
public:
Derived()
{
cout << "Derived()" << endl;
} ~Derived()
{
cout << "~Derived()" << endl;
}
}; int main()
{
Base* p = new Derived();
// ...
delete p; return ;
}
运行打印:
Base()
Derived()
~Derived()
~Base()
可以发现,由于基类的析构函数是虚函数,所以我们delete基类指针时,派生类也跟着调用了析构函数,从而避免了内存泄漏,也能满足使用dynamic_cast强制转换了
一般而言,虚构造函数只有在继承下才会被使用,单个类是不会使用虚构函数的,因为虚函数表会产生额外的空间
注意:构造函数不能成为虚函数,因为虚函数表是在构造函数执行后才会进行初始化
23.C++- 继承的多种方式、显示调用父类构造函数、父子之间的同名函数、virtual虚函数的更多相关文章
- C++入门经典-例8.3-子类显示调用父类构造函数
1:当父类含有带参数的构造函数时,创建子类的时候会调用它吗?答案是通过显示方式才可以调用. 无论创建子类对象时调用的是那种子类构造函数,都会自动调用父类默认构造函数.若想使用父类带参数的构造函数,则需 ...
- 基础3:js实现继承的多种方式
js实现继承的多种方式 1. 原型链继承 function Parent() { this.name = 'xwk' } Parent.prototype.getName = function() { ...
- c++继承构造子类调用父类构造函数的问题及关于容器指针的问题及当容器里储存指针时,记得要手动释放
看下面的一个问题: class Person { private: string name; public: Person(const string& s=""){ nam ...
- 虚函数&纯虚函数&抽象类&虚继承
C++ 虚函数&纯虚函数&抽象类&接口&虚基类 1. 多态 在面向对象语言中,接口的多种不同实现方式即为多态.多态是指,用父类的指针指向子类的实例(对象),然后通过 ...
- C++ 子类继承父类纯虚函数、虚函数和普通函数的区别
C++三大特性:封装.继承.多态,今天给大家好好说说继承的奥妙 1.虚函数: C++的虚函数主要作用是“运行时多态”,父类中提供虚函数的实现,为子类提供默认的函数实现.子类可以重写父类的虚函数实现子类 ...
- python基础之继承实现原理、子类调用父类的方法、封装
继承实现原理 python中的类可以同时继承多个父类,继承的顺序有两种:深度优先和广度优先. 一般来讲,经典类在多继承的情况下会按照深度优先的方式查找,新式类会按照广度优先的方式查找 示例解析: 没有 ...
- Python开发基础-Day20继承实现原理、子类调用父类的方法、封装
继承实现原理 python中的类可以同时继承多个父类,继承的顺序有两种:深度优先和广度优先. 一般来讲,经典类在多继承的情况下会按照深度优先的方式查找,新式类会按照广度优先的方式查找 示例解析: 没有 ...
- C# 构造函数如何调用父类构造函数或其他构造函数
class C : B{ C() : base(5) // call base constructor B(5) { } C(int i) : this() // ca ...
- 在C++中子类继承和调用父类的构造函数方法
构造方法用来初始化类的对象,与父类的其它成员不同,它不能被子类继承(子类可以继承父类所有的成员变量和成员方法,但不继承父类的构造方法).因此,在创建子类对象时,为了初始化从父类继承来的数据成员,系统需 ...
随机推荐
- tomcat服务器怎样远程调试
适合windows系统 1.首先tomcat/bin目录下startup.bat打开最前面添加以下代码: SET CATALINA_OPTS=-server -Xdebug -Xnoagent -Dj ...
- 【转载】sql monitor
来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/27067062/viewspace-2129635/ SQL Monitor Report 1. SQL Monit ...
- Reids学习1 -- 初识Redis
1. Reids和其他类型数据库对比 名称 类型 数据库存储选项 查询类型 附加功能 Redis 使用内存存储的非关系数据库 字符串,列表,集和,散列表,有序集合 每个类型有自己的专属命令,还有批量操 ...
- Asp.Net WebAPI核心对象解析(三)
对于.NET的分布式应用开发,可以供我们选择的技术和框架比较多,例如webservice,.net remoting,MSMQ,WCF等等技术.对于这些技术很多人都不会陌生,即时没有深入的了解,但是肯 ...
- Senparc.Weixin SDK 微信公众号 .NET 开发教程 索引
Senparc.WeixinSDK从一开始就坚持开源的状态,这个过程中得到了许多朋友的认可和支持. 目前SDK已经达到比较稳定的版本,这个过程中我觉得有必要整理一些思路和经验,和大家一起分享.也欢迎大 ...
- jenkins在windows服务器上执行含git push命令的脚本权限不足的解决方法
错误摘要 默认情况下执行脚本是没问题的,但是脚本中含有git push命令就无法执行了 用jenkins部署hexo博客时候遇到的,执行hexo d -g一直阻塞至Build was aborted, ...
- Android 流媒体技术见解
一.技术关键点 1. 音频采集:设置合理的采样参数,包括采样率.通道数.采样深度. 2. 视频采集:手机摄像头采集输出的格式(NV12/NV21/YV12),要注意相互间的转换,以及采集出来图像的方向 ...
- 如何备份/迁移wordpress网站
暴力方法,完全备份网站目录及数据库,上传到新主机 首先,如果是迁移到其他新主机中,需要将新主机的环境配置好,还有对应网站目录和数据库,这些尽量和旧主机一样的设置 接下来,备份旧主机文件,找到网站目录, ...
- webpack严格模式!!!忽略
1. babel5 babel: { options: { blacklist: ["useStrict"], // ... }, // ... } 2. babel6 修改.ba ...
- 第58节:Java中的图形界面编程-GUI
欢迎到我的简书查看我的文集 前言: GUI是图形用户界面,在Java中,图形用户界面我们用GUI表示,而GUI的完整英文为: Graphical User Interface(图形用户接口), 所谓图 ...