C++获取private的变量-偷走private
private提供了对数据的封装,使得private成员只能被类自身的成员函数以及类的友元访问,其他的函数或者类想要访问private成员只能通过该类所提供的set和get的方法进行访问,
或者返回其指针或引用(effective C++中提到过要避免返回对象内部构件的引用,指针,或迭代器。这样会提高封装性,帮助 const 成员函数产生 const 效果,并将悬空句柄产生的可能性降到最低,所以但这个方法并不是特别的好)
但如果你想获得一个类的private成员,但是该类的已经在项目被大量的使用,或者是因为其他的原因,你没有办法添加get和set方法时,又应该如何获得该类的private成员呢?
我总结出了以下几种方法
方法一:重定义
#define private public
示例
A.h定义
#pragma once
class A
{
int j;
public:
A() :i(), j(2)
{ }
template<class Type>
void show(Type t)
{
cout << "Hello" << endl;
}
private:
int i;
};
main.cpp
#include <iostream>
#define private public
#include "A.h"
using namespace std; int main()
{
A a;
cout << a.i << endl;
//cout<<a.j<<endl;无法获得
system("pause");
return ;
}
该方法的优点的是简单但也有不少的缺点
1.如果在类的定义时不指定访问标号关键字(public,protected,private),使用默认的private访问限制,那么该方法就无法达到目的了,比如这里的j就无法获得
2.降低代码的可读性,改变的一个关键字的意义,没有注意到这一点的程序员会产生困扰
3.将所有使用了private访问的标号的成员的访问等级都变成了public,降低了数据的封装性
方法二:模拟内存法
A.h定义
#include <iostream>
#include "A.h"
using namespace std; int main()
{
A a;
void *p = &a;
cout << "j:" << *(int*)p << endl;
cout << "i:" << *((int*)p+)<< endl;// *(int*)((char*)p+4)
system("pause");
return ;
}
C++标准中要求,在同一个访问区域中,成员的排列只需符合较晚出现的成员在类的成员中有较高的地址即可,成员之间可能会因为数据对齐所需,添加一些字节
目前各编译器都是吧一个以上的访问区域连在一起,安装声明的顺序成为一个连续的区域
所以类A的一个对象的内存布局类似于这样:

指针p指向j,将p加上一个int长度或者4个char长度就可以指向i了
但这个方法的缺点也很明显,需要程序员自己对类的内存布局有着较强的了解,考虑到数据对齐,虚函数,不同编译器的实现等等方面
比如以下两种情况
1、数据对齐
A.h定义
#pragma once
class A
{
char j;
public:
A() :i(), j()
{ }
template<class Type>
void show(Type t)
{
cout << "Hello" << endl;
}
private:
int i; };
char j占用了一个byte,而i为了数据对齐,在内存布局上并不是与j紧挨着的,而是隔了3个byte,
所以获得i和j的间隔与上一个一样,只是j的类型变了
#include <iostream>
#include "A.h"
using namespace std; int main()
{
A a;
void *p = &a;
cout << "j:" << *((char*)p) << endl;
cout << "i:" << *((int*)p+)<< endl;
system("pause");
return ;
}
2.加入虚函数
A.h定义
#pragma once
class A
{
int j;
public:
A() :i(), j()
{ }
virtual void show()
{ }
template<class Type>
void show(Type t)
{
cout << "Hello" << endl;
}
private:
int i; };
编译器为了支持虚函数,会在类的每一个对象中,产生一个额外的虚函数指针指向相应的虚函数表,不同的编译器对这个指针处理不同,有点将它放在了类对象的尾端,有的将它放在了类对象的开始处
vs2013将它放在了类的开头处
所以类A的一个对象的内存布局应该类似于这样:

需要将p加上4个字节后才能指向j
#include <iostream>
#include "A.h"
using namespace std; int main()
{
A a;
void *p = &a;
cout << "j:" << *((int*)p+) << endl;
cout << "i:" << *((int*)p+)<< endl;
system("pause");
return ;
}
所以如果虚函数过多,又加入了虚继承, 类里面又有大量程序员自己定义的类型,那么该方法就会很麻烦了。
方法三:李代桃僵
A.h的定义
#pragma once
class A
{
char j;
public:
A() :i(), j('b')
{ }
virtual void show()
{ }
template<class Type>
void show(Type t)
{
cout << "Hello" << endl;
}
private:
int i; };
李代桃僵法是模拟内存布局的另一个实现方式
我们看到现在A里有一个虚函数,一个j和一个i
如果直接使用模拟内存法的话会很麻烦
所以我们可以另声明一个对象B,它的内存布局和A的一样,只是i和j的访问限制变成了public
这样我们可以把一个指向A的对象的指针当做一个指向B的对象指针来使用
#include <iostream>
#include "A.h"
using namespace std; class B
{
public:
char j;
public:
B() :i(), j('b')
{ }
virtual void show()
{ }
/*template<class Type>
void show(Type t)
{
cout << "Hello" << endl;
}*/
public:
int i; };
int main()
{
A a;
B *b = (B*)&a;
cout<<"j:" << b->j << endl;
cout<<"i:" << b->i << endl;
system("pause");
return ;
}
非虚成员函数show放在函数段中,并不在类对象的布局中占用空间,所以有没有show函数都可以
因为B的对象的内存布局与A一样,只是访问限制不同,所以可以利用对B对象的规则去访问A的对象
一个指向B对象的指针实际指向了一个A对象,对B中j和i的访问实际上是对A对象中i和j的访问
该方法模拟内存法容易了很多,但你需要额外声明一个B对象的定义,而且必须要确保B对象的内存布局要与A对象的一致
方法四 特化函数模板法
a.h的定义
#pragma once
class A
{
char j;
public:
A() :i(), j('b')
{ }
virtual void show()
{ }
template<class Type>
void show(Type t)
{
cout << "Hello" << endl;
}
private:
int i; };
这里我们发现A有个函数模板show,所以我们可以利用对函数模板show进行特化的方式合法的获得i和j的public访问权限
#include <iostream>
#include "A.h"
using namespace std; class B
{ };
template<>
void A::show(B b)
{
cout << "j:"<<this->j << endl;
cout << "i:" << this->i << endl;
}
int main()
{
A a;
a.show(B());
system("pause");
return ;
}
该方法合理,简单,但也有缺点就是相应的类必须要有成员模板,并且该模板的访问限制为public才可以
总结
| 方法 | 优点 | 缺点 | 可移植性 |
| 重定义 | 简单 |
1.如果在类的定义时不指定访问标号关键字(public,protected,private),使用默认的private访问限制,那么该方法就无法达到目的了,比如这里的j就无法获得 2.降低代码的可读性,改变的一个关键字的意义,会没有注意到这一点的程序员照成困扰 3.将所有使用了private访问的标号的成员的访问等级都变成了public,降低了数据的封装性 |
中 |
| 模拟内存法 | 无 | 虚函数过多,又加入了虚继承, 类里面又有大量程序员自己定义的类型时,那么该方法就会很麻烦了。需要程序员对内存布局有较深的认识 | 低 |
| 李代桃僵 | 简单,可能在有些人看来比较清楚 | 需要额外声明一个B对象的定义,而且必须要确保B对象的内存布局要与想要访问的A对象的一致 | 中 |
| 特化函数模板法 | 合理,简单 | 相应的类必须要有成员模板,并且该模板的访问限制为public才可以 | 高 |
C++获取private的变量-偷走private的更多相关文章
- 使用typeid(变量或类型).name()来获取常量或变量的类型---gyy整理
使用typeid(变量或类型).name()来获取常量或变量的类型 <typeinfo> 该头文件包含运行时类型识别(在执行时确定数据类型)的类 typeid的使用 typeid操作 ...
- 获取Field成员变量类
位于java.lang.reflect.Field包中 getModifiers() 成员变量修饰符(public.private) getName() 成员变量名字 getType() 成员变量类型 ...
- Path形状获取字符串型变量数据
Path形状获取字符串型变量数据: var path = new Path(); path.Data = Geometry.Parse("M 100,200 C 100,25 400,350 ...
- mysql 获取设置环境变量
mysql 获取环境变量 show global variables; 获取指定环境变量 show global variables like '%timeout'; 设置环境变量 set globa ...
- GetEnvironmentVariable 获取常用系统变量(转)
源:GetEnvironmentVariable 获取常用系统变量 //譬如 WINDIR 表示系统目录系统变量, 以这样获: var s:string; begin s:=GetEnvironmen ...
- 通过System获取java环境变量的路径
通过System获取java环境变量的路径代码为: import java.io.FileNotFoundException; import java.io.FileOutputStream; imp ...
- 在js中,ajax放在for中,ajax获取得到的变量有误
先看代码 for(var i=0;i<tds.length;i++){ mui.ajax(url+'api/client/gifts/isSigned', {data :{ sqId:" ...
- 在c代码中获取用户环境变量
1 extern char ** environ 这是一个字符串数组,最后一个元素是null,即\0. 2 在代码中的使用方法 直接extern char **environ,然后 直接environ ...
- 正则表达式 js 怎么获取匹配的变量
正则表达式 js 怎么获取匹配的变量 $0 ~ $9 refs xgqfrms 2012-2020 www.cnblogs.com 发布文章使用:只允许注册用户才可以访问!
随机推荐
- lua虚拟机概述
何为虚拟机 用于模拟计算机运行的程序.是个中间层,它处于脚本语言和硬件之间的一个程序.每一门脚本语言都会有自己定义的opcode("操作码"),可以理解为这门程序自己定义的&quo ...
- layout/reflow
http://kb.cnblogs.com/page/534571/ http://gent.ilcore.com/2011/03/how-not-to-trigger-layout-in-webki ...
- BZOJ1787_紧急集合_KEY
题目传送门 LCA,对于每一个(x,y,z),两两求LCA得最优解或求出LCA不同于其他两组的那个为最优解. code: /************************************** ...
- CF1039D You Are Given a Tree 根号分治,贪心
CF1039D You Are Given a Tree LG传送门 根号分治好题. 这题可以整体二分,但我太菜了,不会. 根号分治怎么考虑呢?先想想\(n^2\)暴力吧.对于每一个要求的\(k\), ...
- 安装OpenvSwitch (ovs)
简介 搭建SDN环境少不了SDN交换机,SDN交换机跟普通交换机最大的区别就是将普通交换机的数据平面和控制平面相分离,SDN交换机只负责数据的转发,而控制指令则由更上一级的控制器下发. Open vS ...
- 洛谷 P4478 [BJWC2018]上学路线
洛谷 P4478 [BJWC2018]上学路线 原题 神仙题orz,竟然没有1A....容斥+卢卡斯+crt?? 首先用容斥做,记\(f[i][0/1]\)表示到i号点经过了奇数/偶数个点的方案数,因 ...
- USACO Section1.3
section1.2主要包括5道题和1个编程知识介绍.下面对这6部分内容进行学习. Complete Search 这个翻译成枚举搜索或者穷举搜索.主要用于当写代码时间不够用而且不用考虑程序的效率问题 ...
- Merge语句中NULL的陷阱
NULL表示unknown,不确定值,所以任何值(包括null值)和NULL值比较都是不可知的,在on子句,where子句,Merge或case的when子句中,任何值和null比较的结果都是fals ...
- C#_接口与抽象类
.Net提供了接口,这个不同于Class或者Struct的类型定义.接口有些情况,看似和抽象类一样,因此有些人认为在.Net可以完全用接口来替换抽象类.其实不然,接口和抽象类各有长处和缺陷,因此往往在 ...
- Jmeter+Badboy安装使用文档
Jmeter+Badboy安装使用文档 目录 1.jmeter安装 1 2.Jmeter基础使用 3 3. 使用Jmeter进行分布式测试 ...