C++ 在外部访问对象的protected成员的方法

2024-09-06 23:38

本文主要是介绍C++ 在外部访问对象的protected成员的方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

起因

起因在于,今天在写UE4插件时,有一个对象的protected成员我想要访问。这个类没有提供接口来访问那个成员,并且这个类是定义在引擎代码中的而我不想“污染”引擎代码。

不过,我想到这么做或许可以在不改变原有类的定义下访问其中的protected成员:

  1. 定义一个原有类的子类,在其中定义可以访问protected成员的接口。
  2. 想要访问对象的成员时,将原有类类型的指针转变为子类类型的指针,然后就可以执行刚才定义的操作了。

实践

经过尝试,这种操作是没问题的:

#include<iostream>
using namespace std;//原有的类:
class OriginClass
{
public:OriginClass(int InData) { Data = InData; }
protected:int Data;//想要访问的成员
};//为了访问protected成员而特意定义的类
class OriginClass_Access : public OriginClass
{
public:int GetData() { return Data; }//定义的访问成员的方法
};int main()
{//原有的对象:OriginClass* origin = new OriginClass(9);//转换指针:OriginClass_Access* origin_Access = (OriginClass_Access*)origin;//访问成员:cout << origin_Access->GetData() << endl;
}

输出:

9

将一个父类指针直接转换为一个子类指针,看起来是“危险”的。但这里我定义的子类不包含任何额外的字段,只有额外的方法,因此在这里进行的转换是我确保“安全”的。

查阅

网上查阅后发现,这种方法也有其他人想到,在《一种直接访问C++对象的protected成员的通用方法_温研的专栏 (探索OS内核的奥秘)-CSDN博客》这篇博客中甚至将这种方法定义成了一个宏,使得使用变得更加便利。

下面,我也想仿照它的方法,来提炼出通用的宏来方便使用:

提炼为通用的宏

这个宏将会声明一个子类,再声明一个函数用来访问类的protected成员:

#define PROTECTED_MEMBER_ACCESS_FUNCTION_DEFINE(ClassType,MemberName) \
class ClassType##_##MemberName##_Accessor : public ClassType\
{\
public:\typedef decltype(MemberName) MemberType;\MemberType Get() { return MemberName; }\
};\
ClassType##_##MemberName##_Accessor::MemberType Get_##ClassType##_##MemberName(ClassType* origin)\
{\ClassType##_##MemberName##_Accessor* Accessor = (ClassType##_##MemberName##_Accessor*)origin;\return Accessor->Get();\
}\

使用时,先用这个宏来声明函数:

PROTECTED_MEMBER_ACCESS_FUNCTION_DEFINE(OriginClass,Data)

随后,就可以通过函数来访问了:

int main()
{//原有的对象:OriginClass* origin = new OriginClass(9);//访问成员:cout << Get_OriginClass_Data(origin) << endl;
}

输出:

9

完整代码:

#include<iostream>
using namespace std;//这个宏将会声明一个子类,再声明一个函数用来访问一个类的protected成员
#define PROTECTED_MEMBER_ACCESS_FUNCTION_DEFINE(ClassType,MemberName) \
class ClassType##_##MemberName##_Accessor : public ClassType\
{\
public:\typedef decltype(MemberName) MemberType;\MemberType Get() { return MemberName; }\
};\
ClassType##_##MemberName##_Accessor::MemberType Get_##ClassType##_##MemberName(ClassType* origin)\
{\ClassType##_##MemberName##_Accessor* Accessor = (ClassType##_##MemberName##_Accessor*)origin;\return Accessor->Get();\
}\
//原有的类:
class OriginClass
{
public:OriginClass(int InData) { Data = InData; }
protected:int Data;//想要访问的成员
};//定义访问OriginClass的Data成员的函数:
PROTECTED_MEMBER_ACCESS_FUNCTION_DEFINE(OriginClass,Data)int main()
{//原有的对象:OriginClass* origin = new OriginClass(9);//访问成员:cout << Get_OriginClass_Data(origin) << endl;
}

讨论

这么看来,难道C++里的protected权限其实是“形同虚设”吗?

仔细想想也并不是这样,因为上面这种方法依赖于:“原生类”是可继承的,而这实际上是不一定的。

在UE4中,一个模块中的类/函数是需要标记为模块_API宏,其他模块才可以得到在cpp中的定义。这样,如果一个类的构造函数的内容是在cpp中的,则另一个模块是无法继承这个类的,因为并不知道父类的构造函数的定义。

这篇关于C++ 在外部访问对象的protected成员的方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1143443

相关文章

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

Linux系统中查询JDK安装目录的几种常用方法

《Linux系统中查询JDK安装目录的几种常用方法》:本文主要介绍Linux系统中查询JDK安装目录的几种常用方法,方法分别是通过update-alternatives、Java命令、环境变量及目... 目录方法 1:通过update-alternatives查询(推荐)方法 2:检查所有已安装的 JDK方

SQL Server安装时候没有中文选项的解决方法

《SQLServer安装时候没有中文选项的解决方法》用户安装SQLServer时界面全英文,无中文选项,通过修改安装设置中的国家或地区为中文中国,重启安装程序后界面恢复中文,解决了问题,对SQLSe... 你是不是在安装SQL Server时候发现安装界面和别人不同,并且无论如何都没有中文选项?这个问题也

Java Thread中join方法使用举例详解

《JavaThread中join方法使用举例详解》JavaThread中join()方法主要是让调用改方法的thread完成run方法里面的东西后,在执行join()方法后面的代码,这篇文章主要介绍... 目录前言1.join()方法的定义和作用2.join()方法的三个重载版本3.join()方法的工作原

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

在MySQL中实现冷热数据分离的方法及使用场景底层原理解析

《在MySQL中实现冷热数据分离的方法及使用场景底层原理解析》MySQL冷热数据分离通过分表/分区策略、数据归档和索引优化,将频繁访问的热数据与冷数据分开存储,提升查询效率并降低存储成本,适用于高并发... 目录实现冷热数据分离1. 分表策略2. 使用分区表3. 数据归档与迁移在mysql中实现冷热数据分

Spring Boot从main方法到内嵌Tomcat的全过程(自动化流程)

《SpringBoot从main方法到内嵌Tomcat的全过程(自动化流程)》SpringBoot启动始于main方法,创建SpringApplication实例,初始化上下文,准备环境,刷新容器并... 目录1. 入口:main方法2. SpringApplication初始化2.1 构造阶段3. 运行阶

Olingo分析和实践之ODataImpl详细分析(重要方法详解)

《Olingo分析和实践之ODataImpl详细分析(重要方法详解)》ODataImpl.java是ApacheOlingoOData框架的核心工厂类,负责创建序列化器、反序列化器和处理器等组件,... 目录概述主要职责类结构与继承关系核心功能分析1. 序列化器管理2. 反序列化器管理3. 处理器管理重要方

Python错误AttributeError: 'NoneType' object has no attribute问题的彻底解决方法

《Python错误AttributeError:NoneTypeobjecthasnoattribute问题的彻底解决方法》在Python项目开发和调试过程中,经常会碰到这样一个异常信息... 目录问题背景与概述错误解读:AttributeError: 'NoneType' object has no at