【C++深度剖析教程27】多态的概念与意义

今天来学习一个新的概念,多态!!!多态在C++编程中具有重要的地位与意义,是面向对象的一个重要思想!
加qq1126137994一起学习更多技术~

1、问题引入

父类与子类之间具有赋值兼容性;

*子类对象可以当做父类对象使用(赋值兼容性)

  1. 子类对象可以直接赋值给父类对象
  2. 子类对象可以直接初始化父类对象
  3. 父类指针可以直接指向子类对象
  4. 父类引用可以直接引用子类对象

看一个例子程序来理解一下:

#include <iostream>
#include <string>using namespace std;class Parent
{
public:int mi;void add(int i){mi += i;}void add(int a, int b){mi += (a + b);}
};class Child : public Parent
{
public:int mv;void add(int x, int y, int z){mv += (x + y + z);}
};int main()
{Parent p;Child c;p = c;Parent p1(c);Parent& rp = c;Parent* pp = &c;rp.mi = 100;rp.add(5);             // 没有发生同名覆盖?rp.add(10, 10);        // 没有发生同名覆盖?/* 为什么编译不过? */// pp->mv = 1000;      // pp是父类指针,指向子类c,那么c对象就退化为父类对象// pp->add(1, 10, 100); // pp只能访问父类对象,并且可以访问父类与子类同名的对象// 而不用指定作用域。return 0;
}

-当使用父类指针(引用)指向子类对象时

  1. 子类对象退化为父类对象
  2. 只能访问父类中定义的成员
  3. 可以直接访问被子类覆盖的同名成员而不用指定作用域

2、函数重写

  • 子类中可以重定义父类中已经存在的成员函数
  • 这种重定义发生在继承中,叫做函数重写
  • 函数重写是函数同名覆盖的特殊情况

这里写图片描述

那么当函数重写遇上赋值兼容会发生什么?
先来看一个简单的例子:

#include <iostream>
#include <string>using namespace std;class Parent
{
public:int mi;void add(int i){mi += i;}void add(int a, int b){mi += (a + b);}void print(){cout << "I'm Parent." << endl;}
};class Child : public Parent
{
public:int mv;void add(int x, int y, int z){mv += (x + y + z);}void print(){cout << "I'm Child." << endl;}
};void how_to_print(Parent* p)
{p->print();
}int main()
{Parent p;Child c;how_to_print(&p);    // Expected to print: I'm Parent.how_to_print(&c);    // Expected to print: I'm Child.return 0;
}

运行结果为:
I’m Parent.
I’m Parent.

很显然,这个结果并不是我们想要的,我们想要的是执行 how_to_print(&c); 后打印I’m Child.
这就是函数print重写后,遇到赋值兼容的情况。

问题分析:

  • 编译期,间编译器只能根据指针的类型判断所指向的对象
  • 根据赋值兼容,编译器认为父类指针指向的是父类对象
  • 因此,编译结果只可能是调用父类中定义的同名的函数

小结:

  1. 子类对象可以当做父类对象使用(赋值兼容)
  2. 父类指针可以正确的指向子类对象
  3. 父类引用可以正确的代表子类对象
  4. 子类中可以重写父类的成员函数

3、解决办法-多态的概念

  • 父类中被重写的函数依然会继承给子类
  • 子类中被重写的函数会覆盖父类中的同名函数
    这里写图片描述

而面向对象期望的行为是:

  • 根据实际的对象类型调用具体的成员重写函数
  • 父类指针(引用)指向
    *父类对象,则调用父类中定义的函数
    *子类对象,则调用子类中定义的重写函数

要实现这个行为,就需要引出面向对象中的多态的概念:
面向对象中多态的概念:
- 根据实际的对象类型,决定函数调用的具体目标
- 通用的调用语句,在实际的运行中有多种不同的表现形态
这里写图片描述

C++语言直接支持多态的概念

  1. 通过使用virtual关键字对多态进行支持
  2. 被virtual声明的函数被重写后具有多态性
  3. 被virtual声明的函数叫做虚函数

修改上一个程序:

#include <iostream>
#include <string>using namespace std;class Parent
{
public:int mi;virtual void print(){cout << "I'm Parent." << endl;}};class Child : public Parent
{
public:int mv;void print(){cout << "I'm Child." << endl;}
};void how_to_print(Parent* p)
{p->print();
}int main()
{   Parent p;Child c;how_to_print(&p);how_to_print(&c);return 0;
}

运行结果:
I’m Parent.
I’m Child.

可以看出,这是我们想要的结果。

多态的意义:

  1. 在程序运行过程中展现出动态特性
  2. 函数重写必须多态实现,否则没有意义
  3. 多态是面向对象组件化程序设计的基础特性

4、多态在理论中的概念与意义

理论中的概念:

  1. 静态联编
    *在函数编译期间就能确定具体的函数调用
    -如.函数重载
  2. 动态联编
    *在程序实际运行后才能确定函数的具体调用
    -如.函数重写

给个例子说明:

#include <iostream>
#include <string>using namespace std;class Parent
{
public:virtual void func(){cout << "void func()" << endl;}virtual void func(int i){cout << "void func(int i) : " << i << endl;}virtual void func(int i, int j){cout << "void func(int i, int j) : " << "(" << i << ", " << j << ")" << endl;}
};class Child : public Parent
{
public:void func(int i, int j){cout << "void func(int i, int j) : " << i + j << endl;}void func(int i, int j, int k){cout << "void func(int i, int j, int k) : " << i + j + k << endl;}
};void run(Parent* p)
{p->func(1, 2);     // 展现多态的特性// 动态联编
}int main()
{Parent p;p.func();         // 静态联编p.func(1);        // 静态联编p.func(1, 2);     // 静态联编cout << endl;Child c;c.func(1, 2);     // 静态联编cout << endl;run(&p);run(&c);return 0;
}

上述程序的运行结果为:
void func()
void func(int i) : 1
void func(int i, int j) : (1, 2)

void func(int i, int j) : 3

void func(int i, int j) : (1, 2)
void func(int i, int j) : 3

5、拓展训练

下面是一个展现多态,继承的程序,参考教学视频第49课第二个视频!

#include <iostream>
#include <string>using namespace std;class Boss
{
public:int fight(){int ret = 10;cout << "Boss::fight() : " << ret << endl;return ret;}
};class Master
{
public:virtual int eightSwordKill(){int ret = 8;cout << "Master::eightSwordKill() : " << ret << endl;return ret;}
};class NewMaster : public Master
{
public:int eightSwordKill(){int ret = Master::eightSwordKill() * 2;cout << "NewMaster::eightSwordKill() : " << ret << endl;return ret;}
};void field_pk(Master* master, Boss* boss)
{int k = master->eightSwordKill();int b = boss->fight();if( k < b ){cout << "Master is killed..." << endl;}else{cout << "Boss is killed..." << endl;}
}int main()
{Master master;Boss boss;cout << "Master vs Boss" << endl;field_pk(&master, &boss);cout << "NewMaster vs Boss" << endl;NewMaster newMaster;field_pk(&newMaster, &boss);return 0;
}

运行结果为:
Master vs Boss
Master::eightSwordKill() : 8
Boss::fight() : 10
Master is killed…
NewMaster vs Boss
Master::eightSwordKill() : 8
NewMaster::eightSwordKill() : 16
Boss::fight() : 10
Boss is killed…

6、总结

  1. 函数重写只可能发生在父类与子类之间
  2. 根据实际对象的类型调用具体的函数,多态性
  3. virtual关键字是C++中支持多态的唯一方法
  4. 被重写的虚函数可以表现出多态性

想获得各种学习资源以及交流学习的加我:
qq:1126137994
微信:liu1126137994
可以共同交流关于嵌入式,操作系统,C++语言,C语言,数据结构等技术问题。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/423229.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

母亲的革命

做了几个星期思想工作&#xff0c;经过一天的舟车之劳&#xff0c;终于把母亲接到了县城住下来。 按理说&#xff0c;老早就想接母亲出来享享清福&#xff0c;顺便带带金果。可母亲一来放心不下家里的土地&#xff0c;二来怕婆媳之间不好相处&#xff0c;所以都推绝了。 母亲今…

springboot+hbase 集成

项目中使用 phoenix 使用SQL 方式来操作Hbase 数据库&#xff0c;但是遇到一个是&#xff0c;SQL在Dbeaver 中查询速度还可以&#xff0c;但是使用phoenixibatis 后返回结果集数据量20w &#xff0c;速度特别慢&#xff0c;先是考虑用redis方式缓存&#xff0c;但是内存有限&am…

java编程思想学习(3):Java中的private、protected、public和default的区别

&#xff08;1&#xff09;对于public修饰符&#xff0c;它具有最大的访问权限&#xff0c;可以访问任何一个在CLASSPATH下的类、接口、异常等。它往往用于对外的情况&#xff0c;也就是对象或类对外的一种接口的形式。 &#xff08;2&#xff09;对于protected修饰符&#xf…

【Makefile由浅入深完全学习记录2】初识 makefile 的结构

继续学习makefile&#xff0c;希望尽快掌握makefile好在面试中更好的掌握方向&#xff01;加qq1126137994一起学习更多技术&#xff01;&#xff01; 1、回顾 makefile的意义&#xff1a; makefile用于定义源文件之间的依赖关系makefile说明如何编译各个源文件并生成可执行文…

只进ResultSet 不支持请求的操作

使用jdbc操作时 抛出异常 只进ResultSet 不支持请求的操作 那肯定是 在使用resultset的游标操作时调用 resultSet.first () resultSet.last等等方法 你要做的就是设置游标支持滚动操作 如果是statement Statement stmtcon.createStatement&#xff08;ResultSet.TYPE_SCROLL_…

【Makefile由浅入深完全学习记录3】伪目标的引入

只有不停的努力&#xff0c;才能看起来很轻松&#xff01;&#xff01;&#xff01;加qq1126137994共同学习探讨更多技术&#xff01;&#xff01;&#xff01; 1、思考&#xff1a; makefile中的目标究竟是什么&#xff1f; 在默认的情况下&#xff1a; make认为目标对应一…

InnoDB的auto_increment指定值被重置问题

有时候新建的表需要对自增列指定初始值&#xff0c;但是有时候会出现明明指定过的初始值却被重置的现象。下面以一个小实验来说明这个问题&#xff1a;MySQL version:5.1.42 OS:redhat5.3无废话&#xff0c;建张表先&#xff1a;代码CREATETABLEsbtest1 ( id int(10) unsigned…

java ReentrantLock 使用

1.ReentrantLock 简单的使用 private Lock lock new ReentrantLock(); lock.lock();用来获取锁。 lock.unlock();用来释放锁 package com.qey.lock;import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Thr…

【Makefile由浅入深完全学习记录4】变量和不同的赋值方式

今天来学习Makefile中变量的不同的赋值方式&#xff01;努力的意义就是让贫穷不再限制你的想象&#xff01;加qq共同学习交流1126137994 1、概念 makefile中支持程序设计中变量的概念makefile中的变量只代表文本数据&#xff08;字符串&#xff09;makefile中的变量名规则 -…

前端学习(100):float注意点整理1

1只会影响后面得元素 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><meta http-equiv"X-UA-Compatible…

ASP.NET小技巧——回传后保持页面的滚动位置

今天在MSDN上看到一篇文章&#xff0c;关于如何在ASP.NET页面回传后保持当前的滚动&#xff08;垂直&#xff09;位置。这个技巧挺实用的&#xff0c;在此做简单的翻译介绍。默认情况下&#xff0c;ASP.NET页面回传到服务器后&#xff0c;页面会跳回顶部。对于一个内容较多的页…

USB转WIFI无线网卡驱动程序(RT5370驱动程序)的移植记录之一

学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群&#xff1a; 962535112 今天记录我在I.MX6Q平台移植RT5370无线网卡驱动程序的过程&#xff0c;加qq1126137994 微信&#xff1a;liu1126137994 共同学习更多技术&#xff01;…

HTTP和RPC的优缺点

在HTTP和RPC的选择上&#xff0c;可能有些人是迷惑的&#xff0c;主要是因为&#xff0c;有些RPC框架配置复杂&#xff0c;如果走HTTP也能完成同样的功能&#xff0c;那么为什么要选择RPC&#xff0c;而不是更容易上手的HTTP来实现了。 本文主要来阐述HTTP和RPC的异同&#xff…

【Makefile由浅入深完全学习记录5】预定义变量的使用

今天学习Makefile预定义变量的使用&#xff0c;加qq&#xff1a;1126137994.微信&#xff1a;liu1126137994一起学习更多技术&#xff01;&#xff01;&#xff01; 1.预定义变量&#xff1a;自动变量 在Makefile中存在一些预定义过的变量&#xff0c;我们可以直接拿来使用而…

NFC 验证平台搭建

如何把 NFC 挂载到 G7的系统中去&#xff0c;调试一步一步的过程中分别做了什么。 1. 在AHB1上找到1个slave的空挡&#xff0c;ahb-slave8&#xff0c;修改相应AMAB里面的ahb_dcdr.v 文件&#xff0c;给NFC分配基地址空间 0x110050002.在ahb1_top文件当中例化NFC&#xff0c;同…

两个原子操作组合到一块不一定是能保证原子性

1.两个原子操作组合到一块不一定是能保证原子性 ConcurrentLinkedQueue AtomicInteger 两个类都为线程安全的类&#xff0c;但是组合起来并不能保证原子性: public static ConcurrentLinkedQueue concurrentLinkedQueue new ConcurrentLinkedQueue();public static Atomic…

【Makefile由浅入深完全学习记录6】Makefile中变量的高级主题上

抓住基础&#xff0c;学习更多技术&#xff0c;迎接挑战&#xff0c;加qq&#xff1a;1126137994 微信&#xff1a;liu1126137994 一起学习更多技术~ 上一篇文章学习了makefile中的预定义变量的使用&#xff0c;今天来继续学习makefile中的变量&#xff01; 1、变量值的替换 …

micro asyn wininet

http://msdn.microsoft.com/en-us/library/cc185684(VS.85).aspx 状态机 http://support.microsoft.com/kb/224318 How To Control Connection Timeout Value by Creating Second Thread转载于:https://www.cnblogs.com/edward259/archive/2010/04/12/1710305.html

USB转WIFI无线网卡驱动程序(内核自带RT5370驱动程序添加)的移植记录之二

学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群&#xff1a; 962535112 今天记录我在I.MX6Q平台添加内核自带RT5370无线网卡驱动程序的过程&#xff0c;加qq1126137994 微信&#xff1a;liu1126137994 共同学习更多技术&…

MYSQL[30]

MYSQL[30]转载于:https://www.cnblogs.com/motadou/archive/2010/04/13/1710849.html