from://http://blog.csdn.net/xuyuanfan/article/details/9935533

在C++中是没有接口的,要真正实现java中的interface功能,需要使用virtual函数的多态继承机制。这里就细讲一下C++中的virtual关键字的用法。

首先设计3个类,包括book、good_book和bad_book。book为基类,而good_book和bad_book继承于book类。

1、book类:包括一个成员变量name和一个虚成员函数getName

2、good_book类:只有一个成员函数getName

3、bad_book类:只有一个成员函数getName

其中book、good_book和bad_book这3个类的getName是同名同参数列表同返回值的成员函数。

三个类的UML图如下:

三个类的源代码如下:

book类的头文件:book.h

  1. #ifndef _BOOK_H_
  2. #define _BOOK_H_
  3. #include <string>
  4. using namespace std;
  5. class  book{
  6. protected:
  7. string name;
  8. public:
  9. book();
  10. virtual string getName();
  11. };
  12. #endif

book类的源文件:book.cpp

  1. #include "book.h"
  2. book::book()
  3. {
  4. name = "book";
  5. }
  6. string book::getName(){
  7. return name;
  8. }

good_book类的头文件:good_book.h

  1. #ifndef _GOOD_BOOK_H_
  2. #define _GOOD_BOOK_H_
  3. #include "book.h"
  4. using namespace std;
  5. class good_book : public book{
  6. public:
  7. string getName();
  8. };
  9. #endif

good_book类的源文件:good_book.cpp

  1. #include "good_book.h"
  2. string good_book::getName(){
  3. return "good "+name;
  4. }

bad_book类的头文件:bad_book.h

  1. #ifndef _BAD_BOOK_H_
  2. #define _BAD_BOOK_H_
  3. #include "book.h"
  4. using namespace std;
  5. class bad_book : public book{
  6. public:
  7. string getName();
  8. };
  9. #endif

bad_book类的源文件:bad_book.cpp

  1. #include "bad_book.h"
  2. string bad_book::getName(){
  3. return "bad "+name;
  4. }

三个类都设计好了,那现在设计场景(main函数)来运用这三个类,源代码如下:

场景文件:main.cpp

  1. #include <iostream>
  2. #include <string>
  3. #include "book.h"
  4. #include "good_book.h"
  5. #include "bad_book.h"
  6. using namespace std;
  7. int main()
  8. {
  9. int pause;
  10. cout<<"=================================================================="<<endl;
  11. cout<<"使用指向基类的指针bk(指向基类对象book)"<<endl;
  12. book *bk;
  13. bk = new book();
  14. cout<<bk->getName()<<endl;
  15. delete(bk);
  16. cout<<"使用指向基类的指针bk(指向派生类对象good_book)"<<endl;
  17. bk = new good_book();
  18. cout<<bk->getName()<<endl;
  19. delete(bk);
  20. cout<<"使用指向基类的指针bk(指向派生类对象bad_book)"<<endl;
  21. bk = new bad_book();
  22. cout<<bk->getName()<<endl;
  23. delete(bk);
  24. cout<<"=================================================================="<<endl;
  25. cout<<"使用指向派生类的指针bks(指向基类对象book)"<<endl;
  26. cout<<"不可以"<<endl;
  27. //    good_book *bks;
  28. //    bks = new book();
  29. //    cout<<bk->getName()<<endl;
  30. //    delete(bk);
  31. cout<<"使用指向派生类的指针bks(指向派生类对象good_book)"<<endl;
  32. good_book *bks;
  33. bks = new good_book();
  34. cout<<bk->getName()<<endl;
  35. delete(bk);
  36. cout<<"使用指向派生类的指针bks(指向基类的其他派生类对象bad_book)"<<endl;
  37. cout<<"不可以"<<endl;
  38. //    good_book *bks;
  39. //    bks = new bad_book();
  40. //    cout<<bk->getName()<<endl;
  41. //    delete(bk);
  42. cout<<"=================================================================="<<endl;
  43. cout<<"使用基类实例对象Obk"<<endl;
  44. book Obk;
  45. cout<<Obk.getName()<<endl;
  46. cout<<"使用派生类实例对象Ogood_book"<<endl;
  47. good_book Ogood_book;
  48. cout<<Ogood_book.getName()<<endl;
  49. cout<<"使用派生类实例对象Obad_book"<<endl;
  50. bad_book Obad_book;
  51. cout<<Obad_book.getName()<<endl;
  52. cin>>pause;
  53. return 0;
  54. }

编译运行程序,得出结果:

由输出结果可以知道,在两个派生类继承并覆写了基类的虚函数的情况下:

1、使用基类类型指针,那它指向哪个对象实例就会调用哪个对象的函数;

2、使用派生类类型指针,它不能指向基类和该基类的其他派生类,只能指向该派生类对象并调用该派生类对象的函数;

3、使用对象实例,那使用的是哪个对象实例就会调用哪个对象的函数。

在这里,再把该虚函数改为普通函数,即把book.h 文件的getName函数前面的virtual关键字给去掉,源代码如下:

book类的头文件:book.h

  1. #ifndef _BOOK_H_
  2. #define _BOOK_H_
  3. #include <string>
  4. using namespace std;
  5. class  book{
  6. protected:
  7. string name;
  8. public:
  9. book();
  10. string getName();
  11. };
  12. #endif

再次编译运行程序,得出结果:

对照没去掉virtual关键字运行的结果可知道,唯一的区别就是——若使用基类类型指针,那不管它指向哪个对象实例都会调用基类对象的函数;

C++关键字之virtual的更多相关文章

  1. 浅析c++中virtual关键字

    http://blog.csdn.net/djh512/article/details/8973606 1.virtual关键字主要是什么作用? c++中的函数调用默认不适用动态绑定.要触发动态绑定, ...

  2. C#的New关键字的几种用法

    一.在C#中,new这个关键字使用频率非常高,主要有3个功能: a)     作为运算符用来创建一个对象和调用构造函数. b)     作为修饰符. c)      用于在泛型声明中约束可能用作类型参 ...

  3. .NET 关键字

    一.base关键字 可以通过base关键字访问上一级父类方法的访问.静态static函数无法调用base 二.new 关键字new new有2个作用. new运算符   用来分配内存空间和初始化对象. ...

  4. C# 部分关键字

    关键字: virtual:  虚方法,本身可以被实例化,也可以在派生类中重写该方法: override:在派生类重写基类虚方法时声明,避免了C++中的潜在运行错误: abstract:声明为抽象类.抽 ...

  5. c#和java中的方法覆盖——virtual、override、new

    多态和覆盖 多态是面向对象编程中最为重要的概念之一,而覆盖又是体现多态最重要的方面.对于像c#和java这样的面向对象编程的语言来说,实现了在编译时只检查接口是否具备,而不需关心最终的实现,即最终的实 ...

  6. Asp.Net中virtual、override理解

    virtual关键字用于指定属性或方法在派生类中重写.默认情况下,派生类从其基类继承属性和方法,如果继承的属性或方法需要在派生类中有不同的行为,则可以重写它,即可以在派生类中定义该属性或方法的新实现, ...

  7. Delphi 方法:overload、override、virtual、dynamic、abstract

    1.overload 在Pascal语法规则中,同一个UNIT里是不能存在两个同名的函数的,例如: function func(): Boolean; function func(const x: C ...

  8. ASP.NET MVC5+EF6+EasyUI 后台管理系统(59)-BLL层重构

    系列目录 前言:  这应该是本系统最后一次重构,将重构BLL层和Model层.来完全取代代码生成器生成的BLL层和DAL层.完全废掉了代码生成器的DAL,BLL,MODEL层.  全自动生成增,删,改 ...

  9. Atitit 延迟绑定架构法attilax总结

    Atitit 延迟绑定架构法attilax总结 配置文件的延迟绑定1 Api属性与方法的回调延迟绑定1 后期绑定和前期绑定2 延迟调用2 用 Java 语言延迟绑定2 什么是推迟绑定 C++3 配置文 ...

随机推荐

  1. 不借助autolt实现下载文件到指定目录

    今天尝试了下不用借助autolt完成下载文件到指定目录, 好处:在于集成回归,远程机可以绕过执行autolt程序权限问题,导致autolt程序无法调用,不能完成脚本的回归 Firefox浏览器已经成功 ...

  2. django之class Meta

    通过一个内嵌类 "class Meta" 给你的 model 定义元数据, 类似下面这样: class Foo(models.Model): bar = models.CharFi ...

  3. laravel 批量更新

    /** * 转发动态和资讯数量统计 */ public function forwardCounts(FeedModel $feedModel) { //统计动态转发的id $feeds=$feedM ...

  4. vs2010下sort比较函数链接错误问题

    环境:win7 + vs2010 + C++ 实现vector的sort算法,在类的头文件中写入比较函数时会出现链接错误: error LNK2005: "bool __cdecl comp ...

  5. 图学ES6-5.正则的扩展

  6. 上海高校金马五校赛 J - 小Y写文章

    题目大意: 给你n个数字, 定义不连贯值为, max(abs(a[ i ] - b[ i ])) ,现在让你把m个新的数字插入n + 1 个空位中,使得不连贯值最小. 思路:二分不连贯值, 每次进行二 ...

  7. spark优化之临时目录

    官方是这样说的: Directory to use for "scratch" space in Spark, including map output files and RDD ...

  8. 两种思想实现基于jquery的延时导航菜单,可做延时触发器!

    1. 函数式 html如下: <div class="box"> <ul class="clear-fix"> <li class ...

  9. yaml.parser.ParserError

    ERROR: yaml.parser.ParserError: while parsing a block mapping in "./docker-compose.yml", l ...

  10. qq sid qq sid 是什么 qq sid 怎么用

    韩梦飞沙  韩亚飞  313134555@qq.com  yue31313  han_meng_fei_sha ======= qq sid qq sid  是什么 qq sid  怎么用 ===== ...