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

*(void**)解析——如何设计可以在32位下访问到内存区域的前4个字节,在64位下访问到前8个字节?

文章目录

  • \*(int*)
  • \*(void**)
  • 把*(void**)设计成函数,方便调用

  最近在写项目的时候遇到这样一个场景:需要管理多个空闲的内存块,把它们以链表的形式连接起来,那就需要在内存块的头4个字节(32位下)存放下一个内存块的地址。

*(int*)

  内存块的地址是void类型的,我一开始想到的就是把内存块的首地强转成int的类型,然后解引用,这样就能以int对象的方式去访问这个内存块了,也就是访问这个内存块的前4个字节。
  如下代码所示,这里通过malloc申请了两个16字节大小的内存块obj1和obj2,然后想要在obj1内存块的头4个字节存放obj2的地址。

  1. 把obj2的首地址转换成int类型存放在obj1的头四个字节中
  2. 去obj1头4个字节的内容,并强转成void*类型,给到obj1,近似与链表中p = p->next

代码如下:

#include <iostream>
using namespace std;

int main()
{
	void* obj1 = malloc(16);
	void* obj2 = malloc(16);

	cout << "obj1:" << obj1 << endl;
	cout << "obj2:" << obj2 << endl;

	*(int*)obj1 = (int)obj2;  //把obj2的首地址转换成int类型存放在obj1的头四个字节中
	obj1 = (void*)*(int*)obj1;  //去obj1头4个字节的内容,并强转成void*类型,给到obj1,近似与链表中p = p->next

	cout << "obj1:" << obj1 << endl;
	cout << "obj2:" << obj2 << endl;

	return 0;
}

执行结果:

从调试结果和打印结果中我们可以看到,我们确实把obj2的地址存放到了obj1的头四个字节。

在这里插入图片描述

可以看到执行了*(int*)obj1 = (int)obj2;后,obj1的头四个字节被修改为obj2的首地址

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述


  可以看到在32位情况下是没有问题的,但是在64位下一个地址是8字节的,而int是4字节,所以不能使用int。在64位下,我后来改用long long是可以的,于是我写了一个条件编译,如果在32位下就用int,在64位下就用long long。

代码:

#include <iostream>
using namespace std;

#ifdef _WIN64
#define MY_TYPE long long
#elif _WIN32
#define MY_TYPE int
#endif

int main()
{
	void* obj1 = malloc(16);
	void* obj2 = malloc(16);

	cout << "obj1:" << obj1 << endl;
	cout << "obj2:" << obj2 << endl;

	*(MY_TYPE*)obj1 = (MY_TYPE)obj2;
	obj1 = (void*)*(MY_TYPE*)obj1;

	cout << "obj1:" << obj1 << endl;
	cout << "obj2:" << obj2 << endl;

	return 0;
}

运行结果:
在这里插入图片描述
可以看到,运行结果无误。

*(void**)

  虽然但是,条件编译的方法很挫,后来我又发现了一种新的方法,就是把void*的内存块首地址强转成void**类型,然后再解引用,此时访问到的就是一个void*的对象,而void*在32位下是4字节,64位下是8字节,此时通过访问这个void*对象,在32位下就可以访问到内存块的前4个字节,64位下前8个字节,就不需要条件编译了。

代码:

#include <iostream>
using namespace std;

//#ifdef _WIN64
//#define MY_TYPE long long
//#elif _WIN32
//#define MY_TYPE int
//#endif

int main()
{
	void* obj1 = malloc(16);
	void* obj2 = malloc(16);

	cout << "obj1:" << obj1 << endl;
	cout << "obj2:" << obj2 << endl;

	//*(MY_TYPE*)obj1 = (MY_TYPE)obj2;
	//obj1 = (void*)*(MY_TYPE*)obj1;
	//实际上就是把MY_TYPE换成void*,由于内存块地址本身就是void*的,所以不需要强转。
	*(void**)obj1 = obj2;
	obj1 = *(void**)obj1;

	cout << "obj1:" << obj1 << endl;
	cout << "obj2:" << obj2 << endl;

	return 0;
}

运行结果:
在这里插入图片描述

把*(void**)设计成函数,方便调用

  为了方便调用,我把*(void**)设计成了一个函数,如下所示:

static void*& NextObj(void* obj)
{
	return *(void**)obj;
}

  该函数返回的是一个void对象的引用,这样我们就可以通过这个返回值修改void对象中的内容。最终代码如下所示:调用起来就十分方便,使用NextObj(p)就像使用p->next一样

#include <iostream>
using namespace std;

//#ifdef _WIN64
//#define MY_TYPE long long
//#elif _WIN32
//#define MY_TYPE int
//#endif

static void*& NextObj(void* obj)
{
	return *(void**)obj;
}

int main()
{
	void* obj1 = malloc(16);
	void* obj2 = malloc(16);

	cout << "obj1:" << obj1 << endl;
	cout << "obj2:" << obj2 << endl;

	//*(MY_TYPE*)obj1 = (MY_TYPE)obj2;
	//obj1 = (void*)*(MY_TYPE*)obj1;
	//实际上就是把MY_TYPE换成void*,由于内存块地址本身就是void*的,所以不需要强转。
	//*(void**)obj1 = obj2;
	//obj1 = *(void**)obj1;

	NextObj(obj1) = obj2;
	obj1 = NextObj(obj1);

	cout << "obj1:" << obj1 << endl;
	cout << "obj2:" << obj2 << endl;

	return 0;
}

http://www.kler.cn/a/5124.html

相关文章:

  • 基于单片机的无线智能窗帘控制器的设计
  • 深入理解 C 语言中浮点型数据在内存中的存储
  • 设计模式-结构型-组合模式
  • 【llm/ollama/qwen】在本地部署qwen2.5-coder并在vscode中集成使用代码提示功能
  • Elixir语言的学习路线
  • SpringBoot3动态切换数据源
  • 面试官:ThreadLocal了解吗?用过吗?原理是什么?底层数据如何存储的?
  • [oeasy]python0122_日韩字符_日文假名_JIS_Shift_韩国谚文
  • 蓝桥杯刷题冲刺 | 倒计时8天
  • 2023年超全的Android面经(23/30)设计模式安卓源码案例
  • 学术论文等级与分类标准——JCR
  • Element Plus 实例详解(五)___Scrollbar 滚动条
  • 语句【C++】
  • linux创建守护进程
  • Mybatis(一)-------
  • shell 脚本之一键部署安装 Nginx
  • 并查集和哈希表的实现
  • 【Python】线程
  • java企业级信息系统开发学习笔记02初探spring——利用组件注解符精简spring配置文件
  • 第14章_MySQL事务日志
  • 2009年9月Java全国计算机等级考试二级笔试试卷
  • 可用的公开 RTSP/ RTMP 在线视频流资源地址(亲测可行)
  • python去掉字符串中的指定字符的方法
  • 《代码实例》Vue组件与路由
  • 蓝桥杯刷题冲刺 | 倒计时9天
  • 企业安全现状与未来趋势如何?