C++中重载与重写函数区别及虚函数

C++中重载与重写函数区别及虚函数

ID:37909105

大小:39.50 KB

页数:11页

时间:2019-06-02

C++中重载与重写函数区别及虚函数_第1页
C++中重载与重写函数区别及虚函数_第2页
C++中重载与重写函数区别及虚函数_第3页
C++中重载与重写函数区别及虚函数_第4页
C++中重载与重写函数区别及虚函数_第5页
资源描述:

《C++中重载与重写函数区别及虚函数》由会员上传分享,免费在线阅读,更多相关内容在教育资源-天天文库

1、C++中重载与重写函数区别及虚函数C++中的虚函数(virtualfunction) 1.简介    虚函数是C++中用于实现多态(polymorphism)的机制。核心理念就是通过基类访问派生类定义的函数。假设我们有下面的类层次:classA{public:   virtualvoidfoo(){cout<<"A::foo()iscalled"<

2、使用的时候,我们可以:A*a=newB();a->foo();      //在这里,a虽然是指向A的指针,但是被调用的函数(foo)却是B的!   这个例子是虚函数的一个典型应用,通过这个例子,也许你就对虚函数有了一些概念。它虚就虚在所谓“推迟联编”或者“动态联编”上,一个类函数的调用并不是在编译时刻被确定的,而是在运行时刻被确定的。由于编写代码的时候并不能确定被调用的是基类的函数还是哪个派生类的函数,所以被成为“虚”函数。   虚函数只能借助于指针或者引用来达到多态的效果,如果是下面这样的代码,则虽然是虚函数,但它不是

3、多态的:classA{public:   virtualvoidfoo();};classB:publicA{   virtualvoidfoo();};voidbar(){   Aa;   a.foo();  //A::foo()被调用}1.1多态    在了解了虚函数的意思之后,再考虑什么是多态就很容易了。仍然针对上面的类层次,但是使用的方法变的复杂了一些:voidbar(A*a){   a->foo();//被调用的是A::foo()还是B::foo()?}因为foo()是个虚函数,所以在bar这个函数中,只根据这段

4、代码,无从确定这里被调用的是A::foo()还是B::foo(),但是可以肯定的说:如果a指向的是A类的实例,则A::foo()被调用,如果a指向的是B类的实例,则B::foo()被调用。这种同一代码可以产生不同效果的特点,被称为“多态”。1.2多态有什么用?    多态这么神奇,但是能用来做什么呢?这个命题我难以用一两句话概括,一般的C++教程(或者其它面向对象语言的教程)都用一个画图的例子来展示多态的用途,我就不再重复这个例子了,如果你不知道这个例子,随便找本书应该都有介绍。我试图从一个抽象的角度描述一下,回头再结合那

5、个画图的例子,也许你就更容易理解。   在面向对象的编程中,首先会针对数据进行抽象(确定基类)和继承(确定派生类),构成类层次。这个类层次的使用者在使用它们的时候,如果仍然在需要基类的时候写针对基类的代码,在需要派生类的时候写针对派生类的代码,就等于类层次完全暴露在使用者面前。如果这个类层次有任何的改变(增加了新类),都需要使用者“知道”(针对新类写代码)。这样就增加了类层次与其使用者之间的耦合,有人把这种情况列为程序中的“badsmell”之一。   多态可以使程序员脱离这种窘境。再回头看看1.1中的例子,bar()作为

6、A-B这个类层次的使用者,它并不知道这个类层次中有多少个类,每个类都叫什么,但是一样可以很好的工作,当有一个C类从A类派生出来后,bar()也不需要“知道”(修改)。这完全归功于多态--编译器针对虚函数产生了可以在运行时刻确定被调用函数的代码。1.3如何“动态联编”    编译器是如何针对虚函数产生可以再运行时刻确定被调用函数的代码呢?也就是说,虚函数实际上是如何被编译器处理的呢?Lippman在深度探索C++对象模型[1]中的不同章节讲到了几种方式,这里把“标准的”方式简单介绍一下。   我所说的“标准”方式,也就是所谓

7、的“VTABLE”机制。编译器发现一个类中有被声明为virtual的函数,就会为其搞一个虚函数表,也就是VTABLE。VTABLE实际上是一个函数指针的数组,每个虚函数占用这个数组的一个slot。一个类只有一个VTABLE,不管它有多少个实例。派生类有自己的VTABLE,但是派生类的VTABLE与基类的VTABLE有相同的函数排列顺序,同名的虚函数被放在两个数组的相同位置上。在创建类实例的时候,编译器还会在每个实例的内存布局中增加一个vptr字段,该字段指向本类的VTABLE。通过这些手段,编译器在看到一个虚函数调用的时候

8、,就会将这个调用改写,针对1.1中的例子:voidbar(A*a){   a->foo();}会被改写为:voidbar(A*a){   (a->vptr[1])();}   因为派生类和基类的foo()函数具有相同的VTABLE索引,而他们的vptr又指向不同的VTABLE,因此通过这样的方法可以在运

当前文档最多预览五页,下载文档查看全文

此文档下载收益归作者所有

当前文档最多预览五页,下载文档查看全文
温馨提示:
1. 部分包含数学公式或PPT动画的文件,查看预览时可能会显示错乱或异常,文件下载后无此问题,请放心下载。
2. 本文档由用户上传,版权归属用户,天天文库负责整理代发布。如果您对本文档版权有争议请及时联系客服。
3. 下载前请仔细阅读文档内容,确认文档内容符合您的需求后进行下载,若出现内容与标题不符可向本站投诉处理。
4. 下载文档时可能由于网络波动等原因无法下载或下载错误,付费完成后未能成功下载的用户请联系客服处理。