(一)背景引入:承接上篇文章的情况3→设已创建基类指针指向派生类对象(如下)[注意此时只能引用基类成员;而不能访问基类没有,而派生类独有的成员],若基类与派生类有相同函数void g(),该调用哪个?
class A{};
class B:public A{};
int main(){
A *p=new B; //基类指针指向派生类对象
}
(二)实操结论
(1)同名不同参:若基类的成员函数g()与派生类的成员函数g(int x)同名不同参,此时无论基类的g()前是否有virtual,都会调用基类的同名函数! [原理]无virtual时显然调用基类函数。有virtual时→当派生类的虚函数与基类中对应虚函数的参数不同时,派生类的虚函数将丢失虚特性,变为重载函数
(2)同名同参:看基类的相同函数前是否有virtual(即看是否为虚函数),若有virtual则调派生类的同名函数[动态联编],若无virtual则调基类的同名函数[静态联编]
(三)极简代码模型实例
·下面代码已是极简模型,关注不同情况下的输出,非常重要!
【注】当一个成员函数被声明为虚函数后,其派生类中同名函数都自动成为虚函数。故在派生类重新声明该虚函数时,可加virtual,也可不加
#include "bits/stdc++.h"
#include<iostream>
using namespace std;
class A{
public:
virtual void g1(){ //若void g1仍输出A的g1(不过此时A的g1不再是虚函数)
cout<<"A的g1"<<endl;
}
virtual void g2(){
cout<<"A的g2"<<endl;
}
void g3(){
cout<<"A的g3"<<endl;
}
};
class B:public A{
public:
void g1(int x){
cout<<"B的g1"<<endl;
}
void g2(){ //等价于virtual void g().见【注】
cout<<"B的g2"<<endl;
}
virtual void g3(){ //若void g3仍输出A的g3(不过此时B的g3不再是虚函数)
cout<<"B的g3"<<endl;
}
};
int main(){
A *p=new B;
p->g1(); //同名不同参,输出A的g1
p->g2(); //同名同参,基有virtual,输出B的g2
p->g3(); //同名同参,基无virtual,输出A的g3
return 0;
}