C++之同名覆盖、多态
一.同名覆盖引发的问题
父子间的赋值兼容--子类对象可以当作父类对象使用(兼容性)
1.子类对象可以直接赋值给父类对象
2.子类对象可以直接初始化父类对象
3.父类指针可以指向子类对象
4.父类引用可以直接引用子类对象
代码示例
#include
#include
using namespace std;
class Parent
{
public:
int mi;
void add(int i)
{
mi += i;
}
void add(int a, int b)
{
mi += (a + b);
}
};
class Child : public Parent
{
public:
int mv;
void add(int x, int y, int z)
{
mv += (x + y + z);
}
};
int main()
{
Parent p;
Child c;
p = c;
Parent p1(c);
Parent& rp = c;
Parent* pp = &c;
rp.mi = 100;
rp.add(5);
rp.add(10, 10);
pp->mv = 1000;
pp->add(1, 10, 100);
return 0;
}
对该代码进行结果预测:通过之前的学习的同名覆盖,程序会在 rp.add(5); rp.add(10, 10); 进行同名覆盖,且在父类指针指向子类对象时可以进行调用
运行结果
通过程序的运行结果看到,与预测的结果不同,这是因为当使用父类指针(引用)指向子类对象时
1.子类对象退化为父类对象--所以在pp->mv时会出错
2.只能访问父类中定义的成员
3.可以直接访问被子类覆盖的同名成员--所以没发生同名覆盖
特殊的同名函数
1.子类中可以重定义父类中已经存在的成员函数
2.这种重定义发生在继承中,叫做函数重写
3.函数重写是同名覆盖的一种特殊情况
Q:当函数重写遇上赋值兼容会发生什么?
代码示例
#include
#include
using namespace std;
class Parent
{
public:
int mi;
void add(int i)
{
mi += i;
}
void add(int a, int b)
{
mi += (a + b);
}
void print()
{
cout << "I'm Parent." << endl;
}
};
class Child : public Parent
{
public:
int mv;
void add(int x, int y, int z)
{
mv += (x + y + z);
}
void print()
{
cout << "I'm Child." << endl;
}
};
void how_to_print(Parent* p)
{
p->print();
}
int main()
{
Parent p;
Child c;
how_to_print(&p);
how_to_print(&c);
return 0;
}
输出结果
问题分析
1.编译期间,编译器只能根据指针的类型判断所指向的对象
2.根据赋值兼容,编译器认为父类指针指向的是父类对象
3.因此,编译结果只可能是调用
在编译这个函数的时候,编译器不可能知道指针p指向了什么,但是编译器没有理由报错。于是,编译器认为最安全的做法时调用父类的print函数因为父类和子类肯定都有相同的print函数
二.多态的概念和意义
函数重写
1.父类中被重写的函数依然会继承给子类
2.子类中重写的函数将覆盖父类中的函数
3.通过作用域分辨符(::)可以访问父类中的函数
A.面向对象中期待的行为
1.根据实际的对象类型判断如何调用重写函数
2.父类指针指向--a.父类对象调用父类中定义的函数b.子类对象则调用子类中定义的函数
B.面向对象的多态的概念
1.根据实际的对象类型决定函数调用的具体目标
2.同样的调用语句在实际运行时有多种不同的表现形态
C.C++语言中直接支持多态的概念
1.通过使用virtual关键字对多态进行支持
2.被virtual声明的函数被重写后具有多态特性
3.被virtual声明的函数叫做虚函数
#include
#include
using namespace std;
class Parent
{
public:
virtual void print()
{
cout << "I'm Parent." << endl;
}
};
class Child : public Parent
{
public:
void print()
{
cout << "I'm Child." << endl;
}
};
void how_to_print(Parent* p)
{
p->print(); // 展现多态的行为
}
int main()
{
Parent p;
Child c;
how_to_print(&p);
how_to_print(&c);
return 0;
}
运行结果
D.多态的意义
1.在程序的运行过程中展现出多态的特性
2.函数重写必须实现多态,否则没有意义
3.多态时面向对象组件化程序设计的基础特性
静态联编--在程序的编译期间就能确定具体的函数调用
动态联编--在程序实际运行后才能确定具体的函数调用
#include
#include
using namespace std;
class Parent
{
public:无锡好的男科医院 http://mobile.zzchnk.com/
virtual void func()
{
cout << "void func()" << endl;
}
virtual void func(int i)
{
cout << "void func(int i) : " << i << endl;
}
virtual void func(int i, int j)
{
cout << "void func(int i, int j) : " << "(" << i << ", " << j << ")" << endl;
}
};
class Child : public Parent
{
public:
void func(int i, int j)
{
cout << "void func(int i, int j) : " << i + j << endl;
}
void func(int i, int j, int k)
{
cout << "void func(int i, int j, int k) : " << i + j + k << endl;
}
};
void run(Parent* p)
{
p->func(1, 2); // 展现多态的特性
// 动态联编
}
int main()
{
Parent p;
p.func();
p.func(1);
p.func(1, 2);
cout << endl;
Child c;
c.func(1, 2);
cout << endl;
run(&p);
run(&c);
return 0;
}
运行结果
小结
1.函数重写只可能发生在父类与子类之间
2.根据实际对象的类型确定调用的具体函数
3.virtual关键字是C++中支持多态的唯一方式
4.被重写的虚函数表现出多态的特性
C++之同名覆盖、多态的更多相关文章
- C++语法小记---同名覆盖
同名覆盖 子类中的同名成员会覆盖父类中的同名成员,但是在内存中仍然存在,只是无法直接访问,需要加上域名才能访问 子类中的同名函数会覆盖父类中的函数,复写是同名覆盖的一种特殊情况,只要不是多态场景,复写 ...
- 第5月第6天 NSOperation isConcurrent category同名覆盖
1. @implementation AFURLConnectionOperation ... - (BOOL)isConcurrent { return YES; } NSOperation调用st ...
- C++继承中的同名覆盖
1,父子间的冲突是由继承带来的,两个类之间存在了继承的关系,必然的会带来一 些问题,本文要讨论的是父子之间成员变量或成员函数的命名问题: 2,思考: 1,子类中是否可以定义父类中的同名成员? 1,可以 ...
- C++中的重载隐藏覆盖&&JAVA中的重载覆盖&&多态
class 类继承默认是private, struct 默认继承是public C++中的隐藏: 只要派生类中出现和基类一样的函数名,基类中的函数就会被派生类中的函数给隐藏(如果派生类和基类中的函数名 ...
- 虚函数的特点就是执行的时候会下降到子类去执行同名覆盖函数 good
var t: TBitBtn;begin t:=TBitBtn.Create(nil); t.Name:='BitBtn100'; t.parent :=Self; // 这里下断点end; 一路跟踪 ...
- JAVA的覆盖、继承和多态的详细解说.this和super的用法
1. 继承: (1)子类的构造方法一定会调用父类的构造方法. (2)任何子类构造方法第一行肯定是this();或者super();两个择一. this();调用本类的其它构造方法.(传递相应参数调用相 ...
- left join 等连接查询遇到同名字段覆盖问题
可以在查询时给字段赋别名,但是需要注意以下:*的位置要在最前面,放在其他地方都会出错.这种写法同名覆盖的字段还在,然后在*的后面加上别名字段,已经可以满足所有需求了 SELECT *,r.id as ...
- c#和java中的方法覆盖——virtual、override、new
多态和覆盖 多态是面向对象编程中最为重要的概念之一,而覆盖又是体现多态最重要的方面.对于像c#和java这样的面向对象编程的语言来说,实现了在编译时只检查接口是否具备,而不需关心最终的实现,即最终的实 ...
- 虚函数&多态
对于经常被问到的虚函数和多态的问题,发现百度百科回答得十分详细,所以自己在百度百科上的解释进行总结 一.虚函数 (1)虚函数简介:在某基类中声明为virtual并在一个或者多个派生类中被重新定义的成员 ...
随机推荐
- 【软工实践】Beta冲刺(3/5)
链接部分 队名:女生都队 组长博客: 博客链接 作业博客:博客链接 小组内容 恩泽(组长) 过去两天完成了哪些任务 描述 新增数据分析展示等功能API 服务器后端部署,API接口的beta版实现 展示 ...
- _nl_intern_locale_data: Assertion `cnt < (sizeof (_nl_value_type_LC_TIME) / sizeof (_nl_value_type_LC_TIME[0]))' failed
在Ubuntu18上使用交叉编译工具,报这个错.研究之下发现,工具的绝对路径过长,ubuntu18对其优化,修改路径,导致报错. 使用命令:export LC_ALL=C
- 单细胞数据normalization方法 | SCTransform
SCTransform Normalization and variance stabilization of single-cell RNA-seq data using regularized n ...
- Android ADB常用命令使用
Android SDK: adb shell 命令的使用(am.pm.wm.screencap.monkey等) https://blog.csdn.net/xietansheng/article/d ...
- SpringBoot @Autowired中注入静态方法或者静态变量
注:用static去定义一个注入的方法或者配置文件值变量,编译时不会有任何异常,运行时会报空指针. Spring官方不推荐此种方法. 原理: https://www.cnblogs.com/chenf ...
- dotfuscator 在混淆.Net Framework 4.0以上版本的时候报错的解决方法
dotfuscator 在混淆.Net Framework 4.0以上版本的时候报错的解决方法 在混淆的时候报错了,错误描述大致如下: Could not find a compatible vers ...
- (转)IIS windows认证
转自 https://my.oschina.net/u/2551141/blog/2878673 IIS配置Windows认证方式: 1.IIS->>要设置的网站->>身份验证 ...
- python实践项目五:操作剪贴板-pyperclip模块
描述:读取剪贴板的内容,修改该内容,再将修改后的内容重新写进剪贴板 注意:执行程序代码前需保证剪贴板有内容,可复制以下内容来测试: Lists of animals Lists of aquarium ...
- [转帖]Windows 7寿终正寝 为何Windows 10屡被吐槽它却无比经典?
Windows 7寿终正寝 为何Windows 10屡被吐槽它却无比经典? https://www.cnbeta.com/articles/tech/908897.htm 是的,一代经典操作系统Win ...
- 【转帖】漏洞数量242:15,英特尔和AMD CPU谁更安全?
漏洞数量242:15,英特尔和AMD CPU谁更安全? http://www.eetop.cn/cpu_soc/6946340.html 越来越多的用户开始怀疑哪种处理器可以最好地保护他们的计算机,数 ...