函数隐藏

更新时间:2024-06-23 16:39

函数隐藏指不同作用域定义的同名函数之间形成函数隐藏,如派生类的函数屏蔽了与其同名的基类函数,类成员函数屏蔽了与其同名的全局外部函数。

定义

“隐藏”是指派生类的函数屏蔽了与其同名的基类函数,具体规则:

如果派生类的函数与基类的函数同名,但是参数不同。此时,若基类无virtual关键字,基类的函数将被隐藏。(注意别与重载混淆,虽然函数名相同参数不同应称之为重载,但这里不能理解为重载,因为派生类和基类不在同一名字空间作用域内。这里理解为隐藏)  如果派生类的函数与基类的函数同名,但是参数不同。此时,若基类有virtual关键字,基类的函数将被隐式继承到派生类的中。vtable此时派生类vtable中的函数指向基类版本的函数地址。同时这个新的函数版本添加到派生类中,作为派生类的重载版本。但在基类指针实现多态调用函数方法时,这个新的派生类函数版本将会被隐藏。  如果派生类的函数与基类的函数同名,不管函数的参数是不是一样,只要基类函数没有virtual关键字。此时,基类的函数被隐藏。(注意别与覆盖混淆,这里理解为隐藏)。  如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数有virtual关键字。此时,基类的函数不会被“隐藏”。

函数隐藏(屏蔽):如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无virtual关键字,基类的函数将被隐藏(注意别与重载混淆);如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有virtual关键字。此时,基类的函数被隐藏(注意别与覆盖混淆)。总结来说:非override的情况下,派生类对象将屏蔽基类同名函数。

区别举例

VC++深入详解:函数的覆盖和隐藏

1.函数的覆盖

在上一节介绍多态性的时候,我们给出了下面的代码片段:

例2-19

class animal

{

public:

virtual void breathe()

{

}

};

class fish:public animal

{

public:

void breathe()

{

}

};

基类animal的breathe函数前添加了virtual关键字,声明该函数为虚函数。在派生类fish中重写了breathe函数,我们注意到,fish类的breathe函数和animal类的breathe函数完全一样,无论函数名,还是参数列表都是一样的,这称为函数的覆盖(override)。构成函数覆盖的条件为:

n 基类函数必须是虚函数(使用virtual关键字进行声明)。

n 发生覆盖的两个函数要分别位于派生类和基类中。

n 函数名称与参数列表必须完全相同。

由于C++的多态性是通过虚函数来实现的,所以函数的覆盖总是和多态关联在一起。在函数覆盖的情况下,编译器会在运行时根据对象的实际类型来确定要调用的函数。

2.函数的隐藏

我们再看例2-20的代码:

例2-20

class animal

{

public:

void breathe()

{

}

};

class fish:public animal

{

public:

void breathe()

{

}

};

你看出来这段代码和例2-19所示代码的区别了吗?在这段代码中,派生类fish中的breathe函数和基类animal中的breathe 函数也是完全一样的,不同的是breathe函数不是虚函数,这种情况称为函数的隐藏。所谓隐藏,是指派生类中具有与基类同名的函数(不考虑参数列表是否相同),从而在派生类中隐藏了基类的同名函数。

初学者很容易把函数的隐藏与函数的覆盖、重载相混淆,我们看下面两种函数隐藏的情况:

(1)派生类的函数与基类的函数完全相同(函数名和参数列表都相同),只是基类的函数没有使用virtual关键字。此时基类的函数将被隐藏,而不是覆盖(请参照上文讲述的函数覆盖进行比较)。

(2)派生类的函数与基类的函数同名,但参数列表不同,在这种情况下,不管基类的函数声明是否有virtual关键字,基类的函数都将被隐藏。注意这种情况与函数重载的区别,重载发生在同一个类中。

下面我们给出一个例子,以帮助读者更好地理解函数的覆盖和隐藏,代码如例2-21所示。

例2-21

class Base

{

public:

virtual void fn();

};

class Derived : public Base

{

public:

void fn(int);

};

class Derived2 : public Derived

{

public:

void fn();

};

在这个例子中,Derived类的fn(int)函数隐藏了Base类的fn()函数,Derived类fn(int)函数不是虚函数(注意和覆盖相区别)。Derived2类的fn()函数隐藏了Derived类的fn(int)函数,由于Derived2类的fn()函数与Base类的fn ()函数具有同样的函数名和参数列表,因此Derived2类的fn()函数是一个虚函数,覆盖了Base类的fn()函数。注意,在Derived2类中,Base类的fn()函数是不可见的,但这并影响fn函数的覆盖。

当隐藏发生时,如果在派生类的同名函数中想要调用基类的被隐藏函数,可以使用“基类名::函数名(参数)”的语法形式。例如,要在Derived类的fn(int)方法中调用Base类的fn()方法,可以使用Base::fn()语句。

有的读者可能会想,我怎样才能更好地区分覆盖和隐藏呢?实际上只要记住一点:函数的覆盖是发生在派生类与基类之间,两个函数必须完全相同,并且都是虚函数。那么不属于这种情况的,就是隐藏了。

最后,我们再给出一个例子,留给读者思考,代码如例2-22所示(EX09.CPP)。

例2-22

#include

class Base

{

public:

virtual void xfn(int i)

{

}

void yfn(float f)

{

}

void zfn()

{

}

};

class Derived : public Base

{

public:

void xfn(int i) //覆盖了基类的xfn函数

{

}

void yfn(int c) //隐藏了基类的yfn函数

{

}

void zfn() //隐藏了基类的zfn函数

{

}

};

void main()

{

Derived d;

Base *pB=&d;

Derived *pD=&d;

pB->xfn(5);

pD->xfn(5);

pB->yfn(3.14f);

pD->yfn(3.14f);

pB->zfn();

pD->zfn();

}

免责声明
隐私政策
用户协议
目录 22
0{{catalogNumber[index]}}. {{item.title}}
{{item.title}}