C++通过宏定义绕过private访问权限<技巧>

博主看了wangjieest的《C++ 突破私有成员访问限制》 之后
想自己进行尝试用宏定义替换的方式突破访问限制
却发现原作者使用的编译器都是VC2008
而目博主使用的是VC2015
直接使用

#define private public

是无法通过编译器检查的
但在编译器的报错信息中
博主发现了下面的几条

1>d:\sdk1\vc\include\xkeycheck.h(214): warning C4005: “private”: 宏重定义
1>d:\protect\text\text.cpp(4): note: 参见“private”的前一个定义
1>d:\sdk1\vc\include\xkeycheck.h(250): fatal error C1189: #error:  The C++ Standard Library forbids macroizing keywords. Enable warning C4005 to find the forbidden macro.

故去查看了keycheck.h这个头文件
发现原来微软为了防止我们直接这么做
为所有C++保留字都设置了define

bool defined ( string $name )
//检查该名称的常量是否已定义。
//如果该名称的常量已定义,返回 TRUE;未定义则返回 FALSE。

同时进行了EMIT WARNING C4005的宏定义
并在结尾设置了一个#error来终止这种行为

//keycheck.h
  #if defined(alignas)  /* check C++ keywords */ \
    || defined(alignof) \
    || defined(asm) \
    || defined(auto) \
    || defined(bool) \
......
    || defined(final)       /* check "identifiers with special meaning" */ \
    || defined(override)    /* (also known as context-sensitive keywords) */ \
    || defined(carries_dependency)  /* check attribute-tokens */ \
    || defined(noreturn)

   #define alignas EMIT WARNING C4005
   #define alignof EMIT WARNING C4005
   #define asm EMIT WARNING C4005
......
   #define carries_dependency EMIT WARNING C4005
   #define noreturn EMIT WARNING C4005
   #error The C++ Standard Library forbids macroizing keywords. Enable warning C4005 to find the forbidden macro.

所以我们只要这样

#include <xkeycheck.h>
#define private public

就可以突破访问限制
原理是:
先#include xkeycheck.h
再#define private public
如果define 在下文中找到了我们重新定义的private 宏
条件编译就会生效
让我们绕开#error

下面是例子
这里我用了自己之前写的一个链表来做示范:Ilist.h

//1.cpp
#include <xkeycheck.h>
#undef private
#define private public
#include "Ilist.h"
void    main()
{
    Ilist<int>li;
    li._size;
}

值得注意的是这样做只会对这一个cpp文件生效
接下来在这个文件内的所有private修饰符都会失效

//2.cpp
#include "Ilist.h"
void    text()
{
    Ilist<int>  li;
    li._size;//编译器会报错
}

而其他#include “Ilist.h”的文件则不受影响
这是由于#nclude在当前位置展开它所包括的文本
而#宏替换是直到文件末把宏定义的标识符替换并不会改变原版文件
所以在2.cpp展开的是原版文件
故不受其影响
但是注意:千万不要在头文件中这么做,否则所有#include 那个头文件的文件的private 都将被替换,这将是灾难性的事故!
总结:个人感觉C++的封装保护只是在语法层面上的保护,你只要想办法骗过编译器的语法检查,就能访问私有成员.

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
vector<vector<int>>是C++中的一种二维向量容器,它用于存储多行多列的整数数据。 要定义一个vector<vector<int>>变量A并初始化为[[0,1,2],[3,4,5]],可以按照以下步骤进行: 1. 定义一个外层vector容器A:vector<vector<int>> A; 2. 定义一个内层vector容器B,并将元素0、1、2依次添加到B中:vector<int> B; B.push_back(0); B.push_back(1); B.push_back(2); 3. 将B添加到A中:A.push_back(B); 4. 清空B的元素:B.clear(); 5. 重新向B中添加元素3、4、5:B.push_back(3); B.push_back(4); B.push_back(5); 6. 再次将B添加到A中:A.push_back(B); 如果想要通过键盘输入n维矩阵并将其转化为二维向量,可以参考以下代码: #include <iostream> #include <vector> using namespace std; int main() { vector<vector<int>> v; // 定义二维向量v vector<int> temp; // 定义临时向量temp int n, i, j, num; cout << "input the dimension:"; cin >> n; cout << "请输入" << n << "*" << n << "的矩阵" << endl; // 输入元素并存储到二维向量v中 for (i = 0; i < n; i++) { temp.clear(); // 清空temp内元素 for (j = 0; j < n; j++) { cin >> num; temp.push_back(num); } v.push_back(temp); } // 遍历输出二维向量v中的所有元素 cout << "<----- 遍历输出所有元素 ----->" << endl; for (i = 0; i < n; i++) { for (j = 0; j < n; j++) { cout << v[i][j] << " "; } cout << endl; } return 0; } 通过A[row][col]的方式可以访问二维向量A中的某个元素,与访问二维数组的方式类似。例如,如果要访问A中的第2行第3列元素,可以使用A。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值