C++中的关键概念:名字查找与继承
关键概念:名字查找与继承
理解 C++ 中继承层次的关键在于理解如何确定函数调用。确定函数调用遵循以下四个步骤:
1) 首先确定进行函数调用的对象、引用或指针的静态类型。
2) 在该类中查找函数,如果找不到,就在直接基类中查找,如此循着类的继承链往上找,直到找到该函数或者查找完最后一个类。如果不能在类或其相关基类中找到该名字,则调用是错误的。
3) 一旦找到了该名字,就进行常规类型检查,查看如果给定找到的定义,该函数调用是否合法。
4) 假定函数调用合法,编译器就生成代码。如果函数是虚函数且通过引用或指针调用,则编译器生成代码以确定根据对象的动态类型运行哪个函数版本,否则,编译器生成代码直接调用函数。
class A
{
protected:
int m_data;
public:
A(int data=0)
{
m_data = data;
}
int GetData()
{
return doGetData();
}
virtual int doGetData()
{
return m_data;
}
};
class B: public A
{
protected:
int m_data;
public:
B(int data = 1)
{
m_data = data;
}
int doGetData()
{
return m_data;
}
};
class C: public B
{
protected:
int m_data;
public:
C(int data=2)
{
m_data = data;
}
};
class Base
{
public:
virtual int fun()
{
cout << "base: virtual int fun()" << endl;
return 0;
}
};
class D1 : public Base
{
public:
int fun(int)
{
cout << "D1: int fun(int)" << endl;
return 0;
}
int myfun(int, int)
{
cout << "Two int" << endl;
return 0;
}
};
class D2 : public D1
{
public:
int fun(int)
{
cout << "D2: int fun(int)" << endl;
return 0;
}
int fun()
{
cout << "D2: int fun()" << endl;
return 0;
}
int myfun(int, int, int)
{
cout << "Three int" << endl;
return 0;
}
};
int main()
{
C c(10);
cout << c.GetData() << endl;
cout << c.A::GetData() << endl;
cout << c.B::GetData() << endl;
cout << c.C::GetData() << endl;
cout << c.doGetData() << endl;
cout << c.A::doGetData() << endl;
cout << c.B::doGetData() << endl;
cout << c.C::doGetData() << endl;
D2 d2;
//d2.myfun(2,2); //“D2::myfun”: 函数不接受 2 个参数。同一个类中定义的函数才可以重载,继承来的不行。
d2.myfun(2,2,2);
getchar();
return 0;
}
运行结果:
说明:
1)成员函数的重载只能发生在同一个类中所定义的若干函数。不能对从父类继承过来的函数进行重载。因为用子类对象调用“重载函数”的时候,名字查找规则首先在子类中进行,结果找到了具有这个名字的函数,不会再向父类继续查找,所以也谈不上重载。(注释处说明了这个问题)
2)因为D1定义了一个 int fun(int) ,它屏蔽了基类的虚函数 virtual int fun()。从Base继承的虚函数不能通过D1对象(或D1的引用或指针)调用,因为该虚函数被屏蔽了,名字查找规则决定了其在D1中找到了名为fun函数,不会继续向上查找。
3)对于前面11111011的输出,只解释第一个,后面类似:cout << c.GetData() << endl;
本来是要调用C类的GetData(),C中未定义,故调用B中的,但是B中也未定义,故调用A中的GetData(),因为A中的doGetData()是虚函数,所以调用B类中的doGetData(),而B类中的doGetData()返回B::m_data,所以输出1。
浙公网安备 33010602011771号