网站首页 > 基础教程 正文
封装、继承、多态是C++的三大基本特性。封装可以隐藏实现细节,使得代码模块化;继承可以扩展已存在的代码模块(类);封装和继承的目的都是为了"代码重用",多态则是为了实现另一个目的:接口重用。
多态说的简单一些就是"相同的调用产生不同的行为"。这句话具体的涵义在下面会通过例子来解释。
一、虚函数
C++中的多态是通过虚函数来实现的。虚函数的作用是允许子类重新定义父类的成员函数,这种行为称为覆盖或者重写。
(1)如果使用了多态,利用基类的指针指向任意一个子类对象,调用相应的虚函数(可调用到子类重写的父类中定义的虚函数),从而实现动态绑定。
(2)如果没有使用虚函数,那么即使在子类中存在与父类同名的函数,基于指针也只能调用基类的该成员函数,无法调用到子类中被重写的该成员函数。
下面先看一个例子:
#include <iostream> using namespace std; //基类 class Shape { protected: int width, height; public: Shape( int a=0, int b=0) { width = a; height = b; } int area() { cout << "Parent class area :" <<endl; return 0; } }; //子类Rectangle class Rectangle: public Shape { public: Rectangle( int a=0, int b=0):Shape(a, b) { } int area () { cout << "Rectangle class area :" <<endl; return (width * height); } }; //子类Triangle class Triangle: public Shape { public: Triangle( int a=0, int b=0):Shape(a, b) { } int area () { cout << "Triangle class area :" <<endl; return (width * height / 2); } }; int main( ) { Shape *shape; Rectangle rec(10,7); Triangle tri(10,5); shape = &rec; shape->area(); shape = &tri; shape->area(); return 0; }
运行结果为:
可以看到即使是将指针shape指向不同的子类(Rectangle/Triangle),执行shape->area();调用的仍然还是父类Shape的成员函数area();。并没有产生上面所说的"多态"的行为。
修改如下:
#include <iostream> using namespace std; //基类 class Shape { protected: int width, height; public: Shape( int a=0, int b=0) { width = a; height = b; } //使用virtual关键字指定为虚函数 virtual int area() { cout << "Parent class area :" <<endl; return 0; } }; //子类Rectangle class Rectangle: public Shape { public: Rectangle( int a=0, int b=0):Shape(a, b) { } virtual int area () { cout << "Rectangle class area :" <<endl; return (width * height); } }; //子类Triangle class Triangle: public Shape { public: Triangle( int a=0, int b=0):Shape(a, b) { } virtual int area () { cout << "Triangle class area :" <<endl; return (width * height / 2); } }; int main( ) { Shape *shape; Rectangle rec(10,7); Triangle tri(10,5); shape = &rec; shape->area(); shape = &tri; shape->area(); return 0; }
执行结果如下:
相同的调用shape->area();产生了不同的行为,这个现象就是上面所说的多态性。
下面对虚函数给出一个定义:
虚函数是在基类中使用关键字virtual声明的函数。在子类中重写基类中定义的虚函数时,会告诉编译器不要静态链接到该函数,而是在程序运行的任何时候根据指针所指向的对象来选择调用的函数,这种操作被称为"动态绑定或后期绑定"。
二、纯虚函数
- 纯虚函数是在基类中声明的虚函数,它没有具体的定义,要求任何派生类都必须重写纯虚函数
- 包含纯虚函数的类称为抽象类,抽象类不可以进行实例化
- 定义纯虚函数是为了实现一个接口,起到一个规范的作用,它规定继承这个类的程序员必须要重写该纯虚函数
为什么需要纯虚函数:在很多情况下,基类本身生成对象时不合情理的。比如说:动物作为一个基类可以派生出老虎类、猫类、狗类,但是动物类本身实例化对象就不合常理。因此引入了纯虚函数,包含纯虚函数的类称为抽象类,用户不能创建抽象类的实例,只能创建它的派生类的实例。
#include <iostream> using namespace std; class Animal { protected: string name_; public: Animal(string name) { name_ = name; } ~Animal() { } //定义纯虚函数 virtual void ShowMe() = 0; }; class Dog: public Animal { public: Dog(string name): Animal(name) { } ~Dog() { } virtual void ShowMe() { cout << "I am a dog and my name is " << name_ << endl; } }; class Cat: public Animal { public: Cat(string name): Animal(name) { } ~Cat() { } virtual void ShowMe() { cout << "I am a cat and my name is " << name_ << endl; } }; int main() { //error: 抽象类不可以实例化 //Animal* animal = new Animal("动物"); Animal* animal; Dog* dog = new Dog("阿黄"); Cat* cat = new Cat("小黑"); animal = dog; animal->ShowMe(); animal = cat; animal->ShowMe(); return 0; }
运行结果如下:
今天的内容就到这儿了。如果对我的推|文有兴趣,欢迎转|载分|享。也可以推|荐给朋友关|注哦。只推干货,宁缺毋滥。
猜你喜欢
- 2024-11-12 金三银四不跳槽更待何时?安卓开发1年字节5面面经,已成功上岸
- 2024-11-12 C++要学到什么程度才能找到实习? c++学完学什么
- 2024-11-12 C++基础语法梳理:inline 内联函数!虚函数可以是内联函数吗?
- 2024-11-12 C++基类中虚析构函数 c++ 虚析构函数
- 2024-11-12 C和C++代码精粹:C语言和C++有什么区别么?
- 2024-11-12 3个面试C++开发岗位的高频笔试题 c++软件开发面试
- 2024-11-12 一文在手,"类间关系"不再困惑
- 2024-11-12 c++的面试总结 c++面试知识点
- 2024-11-12 C++ 虚函数 实例学习 简单易懂 c++虚函数的使用
- 2024-11-12 C++里面的虚析构函数 虚构造函数与虚析构函数
- 最近发表
- 标签列表
-
- gitpush (61)
- pythonif (68)
- location.href (57)
- tail-f (57)
- pythonifelse (59)
- deletesql (62)
- c++模板 (62)
- css3动画 (57)
- c#event (59)
- linuxgzip (68)
- 字符串连接 (73)
- nginx配置文件详解 (61)
- html标签 (69)
- c++初始化列表 (64)
- exec命令 (59)
- canvasfilltext (58)
- mysqlinnodbmyisam区别 (63)
- arraylistadd (66)
- node教程 (59)
- console.table (62)
- c++time_t (58)
- phpcookie (58)
- mysqldatesub函数 (63)
- window10java环境变量设置 (66)
- c++虚函数和纯虚函数的区别 (66)