当前位置: 首页 > news >正文

记录C++中,子类同名属性并不能完全覆盖父类属性的问题

问题代码:

        首先看一段代码:很简单,就是BBB继承自AAA,然后BBB重写定义了同名属性,然后调用父类AAA的打印函数:

#include <iostream>
using namespace std;class AAA {
public:AAA() {}~AAA() {}void func() {YourVar = 20;}void print() {cout << "YourVar: " << YourVar << endl;}public:int YourVar = 10;};class BBB : public AAA {
public:BBB() {YourVar = 20;}~BBB() {}void print1() {cout << "YourVar: " << YourVar << endl;}void YourMethodOrFunction() {}private:int YourVar;
};int main() {BBB b;b.print();//b.print1();cout << "********" << endl;return 0;
}

打印结果:

        会发现,明明我构造函数里面将YourVar的值改为了20,但是还是打印的10.

        这个问题,属于C++的基础问题,我记得,子类继承父类,如果子类中,定义了同名属性的话,那么父类的属性会被隐藏,访问子类的那个属性,就是子类的,不会是父类的。

        但是如果父类里面,有一个打印的函数print,会发现,这个print函数,生效的范围,只有父类,不包括子类。只有将父类的print函数重写,才能生效在子类中:

解决思路1:重写print

#include <iostream>
using namespace std;class AAA {
public:AAA() {}~AAA() {}void func() {YourVar = 20;}void print() {cout << "YourVar: " << YourVar << endl;}public:int YourVar = 10;};class BBB : public AAA {
public:BBB() {YourVar = 20;}~BBB() {}void print() {cout << "YourVar: " << YourVar << endl;}void YourMethodOrFunction() {}private:int YourVar;
};int main() {BBB b;b.print();cout << "********" << endl;return 0;
}

打印结果:

虽然解决了问题,但是明明我父类中,写了print函数,干嘛还重新写一次?还有意义?

解决思路2:去掉子类的同名属性

#include <iostream>
using namespace std;class AAA {
public:AAA() {}~AAA() {}void func() {YourVar = 20;}void print() {cout << "YourVar: " << YourVar << endl;}public:int YourVar = 10;};class BBB : public AAA {
public:BBB() {YourVar = 20;}~BBB() {}//private:
//	int YourVar;
};int main() {BBB b;b.print();cout << "********" << endl;return 0;
}

但是有时候,我们就是要这么重写,那么该怎么办呢?

解决思路3:将父类地址传递给子类

#include <iostream>
using namespace std;class AAA {
public:AAA() {YourVar = new int;*YourVar = 10;}~AAA() {delete YourVar;}void print() {cout << "YourVar: " << *YourVar << endl;}protected:int *YourVar;
};class BBB : public AAA {
public:BBB() {// 将父类地址传递给子类YourVar = AAA::YourVar;*YourVar = 20;}~BBB() {}private:int *YourVar;
};int main() {BBB b;b.print();cout << "********" << endl;return 0;
}

解决思路4:将子类地址传递给父类:

#include <iostream>
using namespace std;class AAA {
public:AAA() {// YourVar = new int;// *YourVar = 10;}~AAA() {delete YourVar;}void print() {cout << "YourVar: " << *YourVar << endl;}protected:int *YourVar;
};class BBB : public AAA {
public:BBB() {YourVar = new int;// 将子类地址传递给父类AAA::YourVar = YourVar;*YourVar = 20;}~BBB() {}private:int *YourVar;
};int main() {BBB b;b.print();cout << "********" << endl;return 0;
}

总结

1. 面对这样的情况,要么就不要写同名属性(但有时候不能保证自己忘记了)

2. 当以后继承的类层次很多的时候,难免会定义同名属性,但是父类的函数,是不会访问子类的同名属性的,这个是这个问题的关键。虽然在子类中,父类的属性确实是隐藏了,但隐藏不代表消失,也不代表父类的函数中,也将这个属性给隐藏或者替换了。

3. 要么就使用指针,将父类和子类的地址变成一样的地址。

4. 虽然最后一种比较麻烦,但是会惊奇的发现,如果是指针的话,可以实现父类中实现所有操作,子类只管创建对象和使用,子类如果想改实现,根据需要同名替代即可。甚至属性也可以写成函数指针那些,是一种非常灵活的使用方式,是一种上层设计模式的简单体现。

#include <iostream>
using namespace std;class AAA {
public:AAA() {// YourVar = new int;// *YourVar = 10;}~AAA() {delete YourVar;}void print() {cout << "YourVar: " << *YourVar << endl;}protected:int *YourVar;
};class BBB : public AAA {
public:BBB() {YourVar = new int;// 将子类地址传递给父类AAA::YourVar = YourVar;*YourVar = 20;}~BBB() {}void print() {cout << "*********************" << endl;cout << "YourVar: " << *YourVar << endl;cout << "*********************" << endl;}private:int *YourVar;
};int main() {BBB b;b.print();return 0;
}

相关文章:

  • Linux编程3.8 进程-守护进程
  • nodejs中使用@maxmind/geoip2-node 查询地理位置信息
  • MySQL的进阶使用方法
  • 3D开发工具HOOPS如何助力3D项目实现扩展现实技术?
  • Trait与生命周期
  • 学习vue3 第四章(reactive全家桶)
  • playwright自动化项目搭建
  • laravel(源码笔记)服务绑定和解析(依赖注入-反射,控制反转)
  • 【DFS+贪心】第十四届蓝桥杯省赛C++ B组《飞机降落》(C++)
  • wordpress给指定ID分类添加特定的字段
  • 【skimage包如何安装】
  • CentOS7使用Docker部署.net Webapi
  • python云上水果超市的设计与实现flask-django-php-nodejs
  • C/C++代码性能优化——数据结构和算法
  • 云手机为电商提供五大出海优势
  • php的引用
  • 03Go 类型总结
  • emacs初体验
  • Java 实战开发之spring、logback配置及chrome开发神器(六)
  • log4j2输出到kafka
  • sublime配置文件
  • web标准化(下)
  • 从零开始的webpack生活-0x009:FilesLoader装载文件
  • 关于extract.autodesk.io的一些说明
  • 区块链将重新定义世界
  • 三栏布局总结
  • 使用 5W1H 写出高可读的 Git Commit Message
  • 《TCP IP 详解卷1:协议》阅读笔记 - 第六章
  • ​二进制运算符:(与运算)、|(或运算)、~(取反运算)、^(异或运算)、位移运算符​
  • (1)虚拟机的安装与使用,linux系统安装
  • (10)ATF MMU转换表
  • (Redis使用系列) Springboot 在redis中使用BloomFilter布隆过滤器机制 六
  • (第27天)Oracle 数据泵转换分区表
  • (附源码)计算机毕业设计SSM疫情居家隔离服务系统
  • (六)vue-router+UI组件库
  • (三)centos7案例实战—vmware虚拟机硬盘挂载与卸载
  • ./configure,make,make install的作用
  • .[hudsonL@cock.li].mkp勒索病毒数据怎么处理|数据解密恢复
  • .net core 6 集成 elasticsearch 并 使用分词器
  • .NET Micro Framework初体验(二)
  • .NET 线程 Thread 进程 Process、线程池 pool、Invoke、begininvoke、异步回调
  • .Net 中的反射(动态创建类型实例) - Part.4(转自http://www.tracefact.net/CLR-and-Framework/Reflection-Part4.aspx)...
  • .Net 转战 Android 4.4 日常笔记(4)--按钮事件和国际化
  • .NetCore部署微服务(二)
  • .Net程序帮助文档制作
  • .net用HTML开发怎么调试,如何使用ASP.NET MVC在调试中查看控制器生成的html?
  • ?.的用法
  • @GetMapping和@RequestMapping的区别
  • @private @protected @public
  • [2019.3.20]BZOJ4573 [Zjoi2016]大森林
  • [ACM] hdu 1201 18岁生日
  • [acwing周赛复盘] 第 94 场周赛20230311
  • [AIGC codze] Kafka 的 rebalance 机制
  • [AIGC] Redis基础命令集详细介绍
  • [android] 天气app布局练习