一.同名覆盖引发的问题

  父子间的赋值兼容--子类对象可以当作父类对象使用(兼容性)

  1.子类对象可以直接赋值给父类对象

  2.子类对象可以直接初始化父类对象

  3.父类指针可以指向子类对象

  4.父类引用可以直接引用子类对象

  代码示例

  #include

  #include

  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->add(1, 10, 100);

  return 0;

  }

  对该代码进行结果预测:通过之前的学习的同名覆盖,程序会在 rp.add(5); rp.add(10, 10); 进行同名覆盖,且在父类指针指向子类对象时可以进行调用

  运行结果

  通过程序的运行结果看到,与预测的结果不同,这是因为当使用父类指针(引用)指向子类对象时

  1.子类对象退化为父类对象--所以在pp->mv时会出错

  2.只能访问父类中定义的成员

  3.可以直接访问被子类覆盖的同名成员--所以没发生同名覆盖

  特殊的同名函数

  1.子类中可以重定义父类中已经存在的成员函数

  2.这种重定义发生在继承中,叫做函数重写

  3.函数重写是同名覆盖的一种特殊情况

  Q:当函数重写遇上赋值兼容会发生什么?

  代码示例

  #include

  #include

  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);

  how_to_print(&c);

  return 0;

  }

  输出结果

  问题分析

  1.编译期间,编译器只能根据指针的类型判断所指向的对象

  2.根据赋值兼容,编译器认为父类指针指向的是父类对象

  3.因此,编译结果只可能是调用

  在编译这个函数的时候,编译器不可能知道指针p指向了什么,但是编译器没有理由报错。于是,编译器认为最安全的做法时调用父类的print函数因为父类和子类肯定都有相同的print函数

  二.多态的概念和意义

  函数重写

  1.父类中被重写的函数依然会继承给子类

  2.子类中重写的函数将覆盖父类中的函数

  3.通过作用域分辨符(::)可以访问父类中的函数

  A.面向对象中期待的行为

  1.根据实际的对象类型判断如何调用重写函数

  2.父类指针指向--a.父类对象调用父类中定义的函数b.子类对象则调用子类中定义的函数

  B.面向对象的多态的概念

  1.根据实际的对象类型决定函数调用的具体目标

  2.同样的调用语句在实际运行时有多种不同的表现形态

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

  1.通过使用virtual关键字对多态进行支持

  2.被virtual声明的函数被重写后具有多态特性

  3.被virtual声明的函数叫做虚函数

  #include

  #include

  using namespace std;

  class Parent

  {

  public:

  virtual void print()

  {

  cout << "I'm Parent." << endl;

  }

  };

  class Child : public Parent

  {

  public:

  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;

  }

  运行结果

  D.多态的意义

  1.在程序的运行过程中展现出多态的特性

  2.函数重写必须实现多态,否则没有意义

  3.多态时面向对象组件化程序设计的基础特性

  静态联编--在程序的编译期间就能确定具体的函数调用

  动态联编--在程序实际运行后才能确定具体的函数调用

  #include

  #include

  using namespace std;

  class Parent

  {

  public:无锡好的男科医院 http://mobile.zzchnk.com/

  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;

  }

  运行结果

  小结

  1.函数重写只可能发生在父类与子类之间

  2.根据实际对象的类型确定调用的具体函数

  3.virtual关键字是C++中支持多态的唯一方式

  4.被重写的虚函数表现出多态的特性

C++之同名覆盖、多态的更多相关文章

  1. C++语法小记---同名覆盖

    同名覆盖 子类中的同名成员会覆盖父类中的同名成员,但是在内存中仍然存在,只是无法直接访问,需要加上域名才能访问 子类中的同名函数会覆盖父类中的函数,复写是同名覆盖的一种特殊情况,只要不是多态场景,复写 ...

  2. 第5月第6天 NSOperation isConcurrent category同名覆盖

    1. @implementation AFURLConnectionOperation ... - (BOOL)isConcurrent { return YES; } NSOperation调用st ...

  3. C++继承中的同名覆盖

    1,父子间的冲突是由继承带来的,两个类之间存在了继承的关系,必然的会带来一 些问题,本文要讨论的是父子之间成员变量或成员函数的命名问题: 2,思考: 1,子类中是否可以定义父类中的同名成员? 1,可以 ...

  4. C++中的重载隐藏覆盖&&JAVA中的重载覆盖&&多态

    class 类继承默认是private, struct 默认继承是public C++中的隐藏: 只要派生类中出现和基类一样的函数名,基类中的函数就会被派生类中的函数给隐藏(如果派生类和基类中的函数名 ...

  5. 虚函数的特点就是执行的时候会下降到子类去执行同名覆盖函数 good

    var t: TBitBtn;begin t:=TBitBtn.Create(nil); t.Name:='BitBtn100'; t.parent :=Self; // 这里下断点end; 一路跟踪 ...

  6. JAVA的覆盖、继承和多态的详细解说.this和super的用法

    1. 继承: (1)子类的构造方法一定会调用父类的构造方法. (2)任何子类构造方法第一行肯定是this();或者super();两个择一. this();调用本类的其它构造方法.(传递相应参数调用相 ...

  7. left join 等连接查询遇到同名字段覆盖问题

    可以在查询时给字段赋别名,但是需要注意以下:*的位置要在最前面,放在其他地方都会出错.这种写法同名覆盖的字段还在,然后在*的后面加上别名字段,已经可以满足所有需求了 SELECT *,r.id as ...

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

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

  9. 虚函数&多态

    对于经常被问到的虚函数和多态的问题,发现百度百科回答得十分详细,所以自己在百度百科上的解释进行总结 一.虚函数 (1)虚函数简介:在某基类中声明为virtual并在一个或者多个派生类中被重新定义的成员 ...

随机推荐

  1. IDEA文件查找功能失效(ctrl+shift+N)

    由于断电.蓝屏引起的强制关机等情况,会导出IDEA文件查找功能失效,Enter file name窗口输入的文字显示红色,无法查出需要的文件.  解决方法: 可以点击File,选择Invalidate ...

  2. 对异步处理的http接口进行性能测试

    以前对接口做性能测试,接口都是同步处理的,请求之后等待响应结果就知道处理结果了,这样只要看这个接口是否异常,如果无异常无报错记录这个接口的响应时间.TPS等性能指标进行分析就可以了,最近在工作中遇到了 ...

  3. lint-staged那些事儿

    一.工具选型 [预提交工具](https://www.npmtrends.com/lint-staged-vs-pre-commit-vs-pretty-quick) 1.lint-staged 检查 ...

  4. [LeetCode] 47. Permutations II 全排列 II

    Given a collection of numbers that might contain duplicates, return all possible unique permutations ...

  5. nrm -- npm镜像源管理

    nrm nrm(npm registry manager )是npm的镜像源管理工具,有时候国外资源太慢,使用这个就可以快速地在 npm 源间切换 安装nrm 在命令行执行命令,npm install ...

  6. 多核vs多处理器

    多核vs多处理器 多核CPU性能最好,但成本最高:多CPU成本小,便宜,但性能相对较差 线程数=cpu处理器个数 * 一个cpu内的核数[如果有超线程,再乘以超线程数] 多核 CPU 和多个 CPU ...

  7. kali 扫描之burp Suite学习笔记1

    1 安装 2 burs功能图解 3 工具栏详解 4 实战 (1) 网络配置 一台kali 一台msf 网络采用nat nat网络设置方法: 查看路由 配置文件 (2) 代理设置 bur代理设置 浏览器 ...

  8. Python和Java、PHP、C、C#、C++等其他语言的对比?

    1111 1.C语言,它既有高级语言的特点,又具有汇编语言的特点,它是结构式语言.C语言应用指针:可以直接进行靠近硬件的操作,但是C的指针操作不做保护,也给它带来了很多不安全的因素.C++在这方面做了 ...

  9. Windows 下升级 node & npm 到最新版本

    查询 Node 的安装目录where node 升级 Node:在官网下载最新的安装包,直接安装即可.https://nodejs.org/ 升级 npmnpm install -g npm 使用 n ...

  10. react中使用typescript时,error: Property 'setState' does not exist on type 'Home'

    问题描述: 我在react中用typescript时,定义一个Home组件,然后在组件里用setState时会有这样一个报错:(如图)Property 'setState' does not exis ...