在类的继承中,类的私有成员在派生类中是“不可见“的,这种”不可见“是指在派生类的成员函数中,或者通过派生类的对象(指针,引用)不能直接访问它们。但是,不能直接访问并不代表不能访问。在派生类还是能够通过调用基类的共有函数的方式来间接地访问基类的私有成员,包括私有成员变量和私有成员函数。考察如下程序。
#include <iostream>
using namespace std;
class A{
int i;
void privateFunc(){
cout<<"this is a private function of base class"<<endl;
}
public:
A(){i=5;}
int getI(){
return i;
}
void usePrivateFunc(){
privateFunc();
}
};
class B:public A{
public:
void printBaseI(){
cout<<getI()<<endl;
}
void usePrivateFunction(){
usePrivateFunc();
}
};
int main(){
B b;
b.printBaseI();
b.usePrivateFunction();
getchar();
}
程序输出结果:
5
this is a private function of base class
在类B中,由于基类A的成员变量i和成员函数privateFunc()都是私有的,所以在类B的成员函数中无法直接访问到它们。但是,由于类A的公有成员函数getI()可以访问到私有成员变量i,而usePrivateFunction()可以访问私有成员函数privateFunc(),所以在类B中通过调用函数getI()和usePrivateFunc()就可以简介访问基类A中的私有成员。
如果基类中并没有提供访问私有成员的公有函数,那么其私有成员是否“存在“呢?还会不会被继承呢?其实,这些私有成员的确是存在的,而且会被继承,只不过程序员无法通过正常的渠道访问到它们。考察如下程序,通过一种特殊的方式访问了类的私有成员。
#include <iostream>
using namespace std;
class A{
int i;
void privateFunc(){
cout<<"this is a private function of base class"<<endl;
}
public:
A(){i=5;}
};
class B:public A{
public:
void printBaseI(){
int* p=reinterpret_cast<int*>(this);//获取当前对象的首地址
cout<<*p<<endl;
}
void usePrivateFunction(){
void (*func)()=NULL;
_asm{
mov eax,A::privateFunc;
mov func,eax;
}
func();
}
};
int main(){
B b;
b.printBaseI();
b.usePrivateFunction();
}
程序输出结果:
5
this is a private function of base class
(1)虽然类A没有提供访问私有成员变量i的公有方法,但是在类A(以及类A的派生类)对象中个,都包含变量i。
(2)虽然类A并没有提供访问私有成员函数privateFunc()的公有函数,但是在程序代码区依然存有函数privateFunc()的代码,通过内联汇编获取该函数的入口地址,仍然可以顺利调用。
综上所述,类的私有成员一定存在,也一定被继承到派生类中,从大小也可以看出派生类包含了基类的私有成员,读者可自行考证。只不过收到C++语法的限制,在派生类中访问基类的私有成员只能通过间接的方式进行。