相关规则:
如果要在【派生类中重新定义】基类的方法,通常应该将【基类方法】声明为一个虚函数。这样,程序将根据【对象类型】而不是【引用或指针类型】来选择方法版本。
C++规定,当一个成员函数被声明为虚函数后,其派生类中的同名函数都自动成为虚函数。因此在派生类重新声明该虚函数时,可以加virtual,也可以不加,但习惯上一般在每一层声明该函数时都加virtual,使程序更加清晰。
例子1:在未声明为virtual函数的情况下,程序将根据引用类型或指针类型来调用,这并不是多态性行为(使用的是不同类型的指针),没有用到虚函数的功能。
class Base{
public:
void show() { cout << "Base show!" << endl; }
};
class Test : public Base{
public :
void show() { cout << "Test show!" << endl; }
};
int main()
{
Test t;
Base m = t;
t.show(); //Test show!
m.show(); //Base show!
Base & q = t;
q.show(); //Base show!
Base * p = &t;
p->show(); //Base show!
Test * ptr = &t;
ptr->show(); //Test show!
return 0;
}
例子2:在声明为virtual函数的情况下,程序将根据对象类型来调用show();
class Base{
public:
virtual void show() { cout << "Base show!" << endl; }
};
class Test : public Base{
public :
virtual void show() { cout << "Test show!" << endl; } //此处virtual可加可不加
};
int main()
{
Test t;
Base m = t;
t.show(); //Test show!
m.show(); //Base show!
Base & q = t;
q.show(); //Test show!
Base * p = &t;
p->show(); //Test show!
return 0;
}
总结:通过虚函数与指向基类对象的指针变量的配合使用,就能方便地调用同一类族中不同类的同名函数,只要先用基类指针指向即可。
例子3:如果在派生类中没有对基类的虚函数重新定义,则派生类简单地继承其直接基类的虚函数
class Base{
public:
virtual void show() { cout << "Base show!" << endl; }
};
class Test : public Base{
public :
};
int main()
{
Test t;
Base m = t;
t.show(); //Base show!
m.show(); //Base show!
Base & q = t;
q.show(); //Base show!
Base * p = &t;
p->show(); //Base show!
return 0;
}
2、虚函数和函数重载的区别
【函数重载】处理的是【同一层次】上的【同名函数】问题,
【虚函数】处理的是【不同派生层次】上的【同名函数】问题
同一类族的【虚函数】的【首部是相同】的,而函数重载时函数的首部是不同的(参数个数或类型不同)。