C++/Java中继承关系引发的调用关系详解
C++:
这里引用到了 http://blog.csdn.net/haoel/article/details/1948051/ 中的内容,还请提前阅读陈大神的这篇博客后在阅读本篇。
覆盖,实现多态的基础,通过虚函数表来实现,下面这个例子便是覆盖 Override
#include<iostream>
using namespace std;
class Base{
public:
Base(){
cout << "Base::Base" << endl;
}
virtual void f(){
cout << "Base::f" << endl;
}
};
class Derived : public Base{
public:
Derived(){
cout << "Devried::Derived" << endl;
}
void f(){
cout << "Derived::f" << endl;
}
};
int main(){
Base *p = new Derived(); // Base::Base CRCL Derived::Derived
p->f(); // Derived::f
}
如果 Base类f()没有被virtual修饰,如下
#include<iostream>
using namespace std;
class Base{
public:
Base(){
cout << "Base::Base" << endl;
}
void f(){
cout << "Base::f" << endl;
}
};
class Derived : public Base{
public:
Derived(){
cout << "Devried::Derived" << endl;
}
void f(){
cout << "Derived::f" << endl;
}
};
int main(){
Base *p = new Derived();
p->f(); // Base::f
}
如果理解了陈大神博客上那些图,应该不会对这个结果产生意外,这是我画的一个草图,并不对应于真实的内存地址情况,只是为了方便说明概念,下面一排描述的是Derived:

解析:首先p是Base类型的指针,其对于p而言其能只能按照Base类型去运算偏移地址等来访问,所以其不可能访问到Derived中的函数。也就是说父类指针的作用域只有int a , Base::f Base::g这一部分。这样就能理解了。
隐藏,父类指针的访问域是父类部分,子类虽然拥有一份父类函数的拷贝,但如果子类中存在于父类同名的函数,则调用和变量类型一致的函数(即父类变量调用父类函数,子类变量调用子类函数,同时成对方的函数被隐藏)。例:
#include<iostream>
using namespace std;
class Base{
public:
int a;
Base(){
cout << "Base::Base" << endl;
}
void f(float a){
cout << "Base::f" << endl;
}
};
class Derived : public Base{
public:
int d;
Derived(){
cout << "Devried::Derived" << endl;
}
void f(int a){
cout << "Derived::f" << endl;
}
};
int main(){
Derived *p = new Derived();
p->f(3.14f); // Derived::f
}
这里虽然传入f的参数是float型,更适合调用Base::f,但根据同名优先调用子类的原则,实际调用为Derived::f。这就是父类同名方法被隐藏,如果指针类型为父类,则称子类方法被隐藏。
如果子类f定义为 void f() 这里对f传值就会编译不通过。
Java
覆盖,java的情况很简单,只要是子类含有和父类的同名方法,就是覆盖(无论子类的函数是否为abstract)。
public class Test {
public static void main(String[] args){
Base base = new Derived();
base.f(); // Derived::f
}
}
class Base {
public void f() {
System.out.println("Base::f");
}
}
class Derived extends Base {
public void f() {
System.out.println("Derived::f");
}
}
隐藏,Java中也存在隐藏,不过这个隐藏和C++不太一样,子类对象引用可以调用父类的同名函数
public class Test {
public static void main(String[] args){
Derived derived = new Derived();
derived.f(); //Derived::f
derived.f(3.14f); // Base::f
}
}
class Base {
public void f(float a) {
System.out.println("Base::f");
}
}
class Derived extends Base {
public void f(int a) {
System.out.println("Derived::f");
}
}
再讲覆盖,C++父类函数中如果调用了自己的虚成员函数,那么由于这个虚成员函数被覆盖了,所以其实相当于调用了子类的函数。例:
#include<iostream>
using namespace std;
class Base{
public:
int a;
Base(){
cout << "Base::Base" << endl;
}
void f(){
g();
}
virtual void g(){
cout << "Base::g" << endl;
}
};
class Derived : public Base{
public:
int d;
Derived(){
cout << "Devried::Derived" << endl;
}
void g(){
cout << "Derived::g" << endl;
}
};
int main(){
Derived *p = new Derived();
p->f(); // Derived::g
}
Java中类似:
public class Test {
public static void main(String[] args){
Base base = new Derived();
base.f(); //Derived::f
}
}
class Base {
public void f() {
g();
}
public void g(){
System.out.println("Base::g");
}
}
class Derived extends Base {
public void g(){
System.out.println("Derived::g");
}
}
由于想利用这种特性,有人会犯一个错误,在父类的构造函数中调用被覆盖的函数,由于子类在构建时先调用父类的构造函数,此时子类为构造所以不能调到子类的覆盖方法,由此
产生了奇怪的错误。
C++/Java中继承关系引发的调用关系详解的更多相关文章
- JAVA中的四种JSON解析方式详解
JAVA中的四种JSON解析方式详解 我们在日常开发中少不了和JSON数据打交道,那么我们来看看JAVA中常用的JSON解析方式. 1.JSON官方 脱离框架使用 2.GSON 3.FastJSON ...
- java中ReentrantLock类的详细介绍(详解)
博主如果看到请联系小白,小白记不清地址了 简介 ReentrantLock是一个可重入且独占式的锁,它具有与使用synchronized监视器锁相同的基本行为和语义,但与synchronized关键字 ...
- Java中的String,StringBuffer,StringBuilder详解与区别
1.String Java中string类是不可变的,其中在声明的源代码中用的final,所以只能声明一次.所以每次在明面上的改变其实是重新生成一个String对象,指针指向新的String对象.同时 ...
- java中String是对象还是类?详解java中的String
有很多人搞不懂对象和类的定义.比如说java中String到底是对象还是类呢? 有人说String 既可以说是类,也可以说是对象. 其实他这么说也没问题, 类和对象其实都是一个抽象的概念. 我们可以把 ...
- 线程:Java中wait、notify、notifyAll使用详解
基础知识 首先我们需要知道,这几个都是Object对象的方法.换言之,Java中所有的对象都有这些方法. public final native void notify(); public final ...
- java中Timer类的详细介绍(详解)
一.概念 定时计划任务功能在Java中主要使用的就是Timer对象,它在内部使用多线程的方式进行处理,所以它和多线程技术还是有非常大的关联的.在JDK中Timer类主要负责计划任务的功能,也就是在指定 ...
- java中Dateformat类的详细使用(详解)
DateFormat其本身是一个抽象类,SimpleDateFormat 类是DateFormat类的子类,一般情况下来讲DateFormat类很少会直接使用,而都使用SimpleDateFormat ...
- Java中的双重检查(Double-Check)详解
在 Effecitve Java 一书的第 48 条中提到了双重检查模式,并指出这种模式在 Java 中通常并不适用.该模式的结构如下所示: ? 1 2 3 4 5 6 7 8 9 10 public ...
- Java中的宏变量,宏替换详解。
群友在微信群讨论的一个话题,有点意思,特拿出来分享一下. 首先来看下面这段程序,和群友分享的大致一样. public static void main(String[] args) { String ...
随机推荐
- ODPS基础
遇到一个项目需求是统计128张分库分表的数据表记录的最大id,通过单表查询计算非常费时,也无法应对分表数更多的情况,因此考虑到通过odps进行任务发布和运算 在云端 http://d2.alibaba ...
- 仿知乎日报App
1.6.Error:Execution failed for task ':app:buildInfoDebugLoader'.> Exception while doing past iter ...
- Delegate(代理)异常:该委托必须有一个目标
转自 Delegate(代理)异常:该委托必须有一个目标 在代理调用BeginInvoke(new AsyncCallback(callBack), null);时,会抛这个异常的原因是该代理变量代理 ...
- System.Drawing.Text.TextRenderingHint 的几种效果
; i < ; i++) { g5.TextRenderingHint = (System.Drawing.Text.TextRenderingHint)i; string txt; ; txt ...
- Delphi IOS class_addMethod
class_addMethod 学习FMX.Platform.iOS.pas文件的处理办法 d:\program files (x86)\embarcadero\studio\17.0\source\ ...
- Unknown picture file extension
Image1.Picture.LoadFromFile('aaa.jpg'); Project Project1.exe raised exception class EInvalidGraphic ...
- C# 6 与 .NET Core 1.0 高级编程 - C# 6 的新功能
个人原创译文,转载请注明出处.有不对的地方欢迎指出与交流. 英文原文:Professional C# 6 and .NET Core 1.0 - What’s New in C# 6 C# 6 的新功 ...
- C++提高编译与链接速度的资料
1,https://blog.csdn.net/lihao21/article/details/47610309 2,https://www.zhihu.com/question/37330979 3 ...
- sortingOrder,sortingLayer
sortingOrder 是sortingLayer内的分级 sortingLayer是抽象的分层,用于决定2D物体绘制的先后顺序. 2D物体分两类:sprite和UI. sprite虽是2D,却可以 ...
- PHP 数组中出现中文乱码,json_encode返回结果为null 或false
想要解决这个问题,没有特别方便的方法,只有循环数组,将数组中的key和value字符串转码,转换为utf-8,即可解决问题. 代码示例: