在C++教材的前几章就介绍了其特性,即:C++是一门面向对象语言,具有封装、继承和多态三个特点。后来,随着编码的增多以及工作经验的积累,对个概念的理解越来越深。编码习惯也严格按照相应的规则,该封装的时候进行封装,该继承的时候进行继承,以使得编程思维从之前的面向过程逐步过渡到面向对象。
作为开发人员,遵循编程规则本来就无可厚非,但是如果大家都遵循规则难免会有创新或者技术进步。有时候,在做某件事或者看到某个实现方案的时候,想想为什么要这么做,有没有更好的实现方案,这个编程或者做事习惯往往使得自己受益匪浅。
比如,我们都知道每个线程都有一个自己的栈,线程内的局部变量出了作用域就会被释放,那么有没有可能跨线程从另外一个线程去访问该线程的局部变量呢?其实,问题不算难,我们只需要尝试即可,但往往缺少的就是这种尝试。对于C++三大特性中的封装特性,如果直接访问私有变量,则编译器会报错,那么有没有其它方式可以访问私有变量呢?
今天,不妨试着反其道而行,尝试以其他方式破坏封装性,直接访问私有变量。
从一段代码说起
代码示例如下:
#includeclassA{ public: A()=default; private: intdata_=0; }; intmain(){ Aa; std::cout<< a.data_ << std::endl; return 0; }
在gcc5.4下进行编译,不出所料,编译失败,报错如下:
test.cc:在函数‘intmain()’中: test.cc:7:15:错误:‘int A::data_’是私有的 intdata_=0;
从报错信息看,因为data_成员变量是私有的,而通过对象访问私有成员变量是不被允许的,除了通过重新定义一个公共接口,在该接口内对data_进行访问外,但是这种方式并没有实现本文的目的即破坏封装性,那么有没有其它方式呢?
第一次尝试
c++标准中有这样一段描述:
The usual access checking rules do not apply to non-dependent names used to specify template arguments of the simple-template-id of the partial specialization. [ Note:The template arguments may be private types or objects that would normally not be accessible. Dependent names cannot be checked when declaring the partial specialization, but will be checked when substituting into the partial specialization. — end note ]
也就是说模板参数可以是某个类的私有类型,所以,我们可以借助此条标准继续实现我们的目的,代码如下:
#includeclassA{ public: A()=default; private: intdata_=0; }; template int&GetPrivateData(A&obj){ returnobj.*Member; } templateint&GetPrivateData<&A::data_>(A&); intmain(){ Aobj; GetPrivateData<&A::data>(obj); return0; }
在上述代码中,定义了一个函数模板,其模板参数为int A::*Member,功能是返回类A中的成员变量,编译后,报错如下:
test.cc:在函数‘intmain()’中: test.cc:7:15:错误:‘int A::data_’是私有的 intdata_=0; ^ test.cc:22:3:错误:在此上下文中 GetPrivateData<&A::data_>(obj);
看来此方式还是行不通,只能另想它法。
第二次尝试
在上面的提示中,显示不能直接访问私有成员,标准提供了个方法,就是将需要访问类私有成员的函数或者类声明为friend。看到这块,你可能会想,有了friend用得着你教?。
很快写出如下这种代码:
classA{ public: A()=default; private: intdata_=0; friendintAccess(constA&a){ returna.data_; } }; intmain(){ Aa; Access(a); return0; }
无疑,上面这种代码可以访问私有成员,但缺点是需要更改类实现,下面将介绍一种方式,其在不修改类本身定义的情况下实现访问私有成员变量。
本着大方向不变的原则,依然使用模板的方式访问私有成员,而对于上节中提示的非法访问私有成员,我也采用将对应函数声明为friend的方式。
#includeclassA{ public: A()=default; private: intdata_=0; }; template< int A::*Member > classAccess{ public: friendintGetPrivateData(A&obj){ returnobj.*Member; } }; templateclassAccess<&A::data_>; intGetPrivateData(A&); intmain(){ Aobj; GetPrivateData(obj); return0; }
编译 & 运行,OK!!!
另辟蹊径
在上一节实现中,使用了friend进行访问控制,所以在考虑有没有不使用friend的方式,于是在网上进行搜索查阅,如下:
classA{ public: A(intnum):data_(num){}; private: intdata_=0; }; templateclassAccess{ public: inlinestaticPtrTypeptr; }; template structPtrTaker{ structTransferer{ Transferer(){ Access ::ptr=T; } }; inlinestaticTransferertr; }; templateclassPtrTaker<&A::data_>;//显示实例化 intmain(){ Aa{10}; intb=a.*Access ::ptr; return0; }
说真的,看到这种实现方式的时候,一脸懵逼,尤其是对模板用的不多的情况下,阅读这短短几十行代码用了一天时间,其间也跟@Chunel骏哥哥一起讨论,奈何太挫了,只能硬着头皮自己研究,也跟群里的大佬们一起讨论了下,再结合自己的理解,分析下这块:
1、因为用到了inline 变量以及模板参数为auto,所以上述代码在cpp17上才可以运行。
2、以&A::data_作为模板参数,对类模板PtrTaker进行显示实例化,在显示实例化的时候,虽然不创建对象,但是对于其中存在的静态变量依然会进行初始化。因此会调用Transferer类的构造函数,从而对Access::ptr进行初始化
看上述代码的时候,一开始卡在了a.*Access
•p = Access
•a.*p
看了下面的代码示例,相信能便于理解:
classData{ public: intnum_=0; }; intmain(){ intData::*ptr=&Data::num_; Datadata; data.*ptr=10; return0; }
好了,我们接着进行讨论。
在使用对象访问成员的时候,其地址实际上分为两部分的,以a.data_为例(此处忽略访问控制权限),一部分是a的this指针,另一部分是data_成员在A结构里的偏移量,这个偏移量存储在&A::data_中。在上面的代码中,这个偏移量存储在静态数据ptr里了,即上面提到的Access::ptr。
所以,a.*p相当于如下:
intA::*p=&A::data_; intoffset=*(longlong*)&p; intdata=*(int*)((char*)&a+offset);
好了,截止为此,通过模板方式访问类私有成员的讨论结束了。
可能有人会有疑问,如果类有多个成员变量,又该如何访问呢,方式类似,代码如下:
#include#include classA{ public: A(intnum,std::stringv):data_(num),value_(v){}; private: intdata_=0; std::stringvalue_; }; template classAccess{ public: inlinestatictypenameTag::typeptr; }; template structPtrTaker{ structTransferer{ Transferer(){ Access ::ptr=V; } }; inlinestaticTransferertr; }; structTag1{ usingtype=intA::*; }; structTag2{ usingtype=std::stringA::*; }; templateclassPtrTaker ;//显示实例化 templateclassPtrTaker ;//显示实例化 intmain(){ Aa{0,"abc"}; std::cout<< "123 " << a.*Access ::ptr; }
审核编辑:刘清
-
C语言
+关注
关注
180文章
7608浏览量
137152 -
编译器
+关注
关注
1文章
1636浏览量
49174 -
C++语言
+关注
关注
0文章
147浏览量
7009 -
gcc编译器
+关注
关注
0文章
78浏览量
3402
原文标题:访问私有成员——从技术实现的角度破坏"封装" 性
文章出处:【微信号:CPP开发者,微信公众号:CPP开发者】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
评论