虚函数这一部分刚开始看的时候也是一头雾水。早上一边看一边pass,好几次有回头去看的冲动。下午找了百度百科,和问了健辉几个问题,才大概有了了解。百科说的很清楚,甚至说了虚函数是怎么动态工作的,剖析很深入、这是它的连接
http://baike.baidu.com/view/161302.htm。以下是我的笔记。
1、除了构造函数外,任意非static成员函数都可以是虚函数。
2、保留字virtual值在类内部的成员函数声明中出现。
3、protected成员不能被类的用户访问;
protected成员可被该类的派生类访问。
派生类只能通过炮声类对象访问其基类的protected成员,派生类对其基类类型对象的protected成员没有访问权限。
4、派生类中虚函数的声明必须与基类中的定义方式完全匹配。但派生类中的虚函数可以返回基类函数所返回类型的派生类的引用(或指针)
5、基类本身可以是一个派生类。
6、运行时确定virtual函数的调用。编译时确定非virtual调用。
7、派生类虚函数调用基类版本时,必须显示使用作用于操作符。如果派生类函数忽略了这样做,则函数调用会在运行时确定并且将是一个自身调用,从而导致无穷递归。
8、#include<iostream>
using namespace std;
class A
{
public:
int n;
virtual void print()
{
cout<<"A"<<endl;
}
};
class B:public A
{ public:
int ret1()
{
return 1;
}
void print()
{
cout<<"B"<<endl;
}
};
int main ()
{
A a;
B b;
A *pb=&b;
a.print();//打印“A”
b.print(); //打印“B”
(*pb).print();//打印“A”A类中的print()函数前加virtual变成虚函数后打印B;
b.A::print();//打印“A”
cout<< b.ret1()<<endl;//输出1
//cout<<(*pb).ret1()<<endl;//编译错误 ,不能访问ret1函数。
system("pause");
return 0;
}
令人迷惑的隐藏规则
本来仅仅区别重载与覆盖并不算困难,但是 C++ 的隐藏规则使问题复杂性陡然增加。这里“隐藏”是指派生类的函数屏蔽了与其同名的基类函数,规则如下:
(1 )如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无virtual关键字,基类的函数将被隐藏(注意别与重载混淆)。
(2 )如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有virtual
关键字。此时,基类的函数被隐藏(注意别与覆盖混淆)。
//---------------------------------------------------------------------------------------
重载与覆盖
成员函数被重载的特征:
(1 )相同的范围(在同一个类中);
(2 )函数名字相同;
(3 )参数不同;
(4 )virtual 关键字可有可无。
覆盖是指派生类函数覆盖基类函数,特征是:
(1 )不同的范围(分别位于派生类与基类);
(2 )函数名字相同;
(3 )参数相同;
(4 )基类函数必须有virtual 关键字。
//---------------------------------------------------------------------------------------