C语言数组和指针,内存之间的关系

 更新时间:2022年2月3日 15:52  点击:449 作者:伴你永居我忆i  

首先论证一维数组和一级指针之前的关系,我们常常使用一级指针指针的方式访问一维数组,只有对内存的理解到位才能理解它们直接的关系。

  • 1.数组名是数组的首地址
  • 2.对数组名取地址得到的还是数组的首地址
  • 3.数组的访问方式其实就是首地址+偏移的寻址访问

我们在程序中会定义很多变量,有基本类型和自定义类型
在进行开发的时候我对内存的访问访问就是通过变量名赋值的方式读写内存
但是如果你看到的直接变量的符号名你将不可能理解内存。
每一种类型都有字节宽度,
char 1字节 short 2字节 int 字节float 4 double 8,其他的自定义类型也有一个对应的大小,对于通过变量名读写操作内存,我们需要自动在脑子里面形成一个映射关系。

int a=10;// 往某段内存地址位x的内存 写入4个字节的数据10
intp =(int)20; //往某段内存地址位x1的内存 写入4个字节的数据20
int** p =(int**)30;// 往某段内存地址位x2的内存 写入4个字节的数据30
int b = a; //读取某段内存地址为x的内存 读取4给字节数据值 写入首地址为b的某段内存 宽度为4,

对于赋值读写需要脑子里面自动分割成一块块内存然后将变量符号名字与对于的首地址(还有宽度)对于起来,忽略数据类型的概念,对于一个变量只关注首地址+数据宽度。

对于使用指针访问数组如图下所示
基本上有点基础的都可以看出来使用直接使用数组arr和使用指针p相比少了
arr是首地址+偏移 然后直接往这个地址里面读或者写
指针的操作方式首先需要得到p对象 所占的空间里面存放的值(这里是数组首地址)然后再通过存放的值+偏移的方式 读写内存
int* p =arr; 把p这个对象 赋值数组首地址 任何指针类型32位下4个字节 64位下8个字节,p这个对象需要4个字节的空间存储arr值,因为p是一个变量,占4个字节 int* p=arr;就是 往p所占的4个字节里面存储arr值首地址
而通过p[_i]下标访问的时候 首先就需要 取出来p里面存的四个字节值y
然后y+偏移的方式 访问读或者写该指针指向数组中的内容

从上面的结果我们可以得到,别看到一级指针操作一维数组 和数组直接访问问之前只多了一层指针变量的寻址,但是含义就完全的变了。
如有些书上说数组作为实参会退化为指针
lea eax,[arr] 是数组首地址保存到eax寄存器中
push eax 参数入栈 此时esp寄存器的值减4,push eax相当与啥?相当于创建了一个零时对象 占4个字节的指针,传递过去,首地址值访问一块4个字节的内存后这段4个字节的内存后编译器认为这是一个指针,所以funtion里面使用sizeof可以看到占4个字节(可以私下去测试),所以把编译器给的类型去掉后 。
int a = (int)arr; 往a占的4个字节内存中写入数组首地址,从汇编的角度来看,是不是找到一丝熟悉的感觉了,没错下面这两个除了编译器附加的一些类型的限制之外没有区别,甚至可以说是完全一样的,我们完全可以使用一个int a;操作任何基本类型和抽象类型对象

int a= (int)arr;
int* p =arr;

函数传递一级指针和二级指针
下图可以看到传递一级是将p的地址ebp-0x3c里面存储的内容 push到栈中传参
而传递二级指针的时候是lea 得到ebp-0x3c这个值 p的地址传递进去
所以对于二级指针而言 我们很容易在funtion2函数里面改变被调用函数里面p指向的内容
(这种方式多与一些库的设计 如ffmpeg 的一些函数设计传递一个一级指针的地址进去(指向NULL)函数里面可以创建某些对象然后然后改变被调函数的指针,释放的时候同理,这样可以减少开发者所作的工作

可以用一级指针指针引用一位数组,以前刚学习的时候我自认而然的使用二级指针引用二维数组,发现是不可以。
进程的内存就是线性的32位可以寻址00000000-0xffffffff 4gb,从内存的角度看没有二维数组的概念,也没有多维数组的概念,对于任何一个变量,关注的应该都是首地址,宽度
int arr[4][3] ; 在内存布局上可以 int arr[12];是完全一样的没有任何区别
二维数组或者多维数组的设计知识为了理解的方便,比如使用二维数组可以更加直观的表示一个nm地图的状态,使用一维度的话没有那么的直观。三维数组可以让我们更直观的表示一个空间的概念,其实它们内存也就是一个一线性的一维数组
int arr[4][3] 我们通过 arr[_i][_j]的访问方式
mov eax,[_i];首先得到_i的值保存到eax寄存器中
lea ecx,arrtow[eax4]; lea指令是得到[]里面的地址 arrtow首地址 + _i41(char占1个字节如果是其他类型需要对应大小)
mov edx ,[_j] ;edx寄存器保存_j的值
movsx eax.byte ptr[ecx+edx] 就是arrtow+_i41+_j1 取出这个内存编号中的值取一个字节放入eax寄存器中 char到int有一个转换
mov [a],eax 放入a变量所在的地址中 四个字节
可以看到二维数组其实也是一个一维数组 首地址+_im宽度+j*宽度的寻址方式
三维数组同理

二级指针不能引用二维数组是因为是因为二级指针操作内存的方式和二维数组的完全不同,一级指针可以引用一维数组是因为它们操作内存的方式是相同的

lea eax,[arrtow]
mov dword ptr [pp],eax //这两行把char** pp = (char**)arrtow arrtow值给 pp变量
mov eax,dword ptr [_i]; //_i的值保存在eax寄存器中
mov ecx.dword ptd[pp]; //得到pp中存储的值 就是arrtow值(数组首地址)
mov edx,dword ptr [ecx+eax4] // 数组首地址+_i4的值 指针宽度占4个字节, 将arrtow+_i 4位首地址读取四个字节 到edx寄存器 (这里就寻址了一次)
mov eax.dword ptr [_j]; //读取_j的值存放在eax寄存器中
movsx ecx,byte ptr [edx+eax]//把在arrtow+_i4地址处取的四个字节数据当做首地址 +_j再次当作首地址取一个字节 当如ecx寄存器
mov dword ptr [a],ecx, 放入a变量中

通过二级指针和二维数组访问内存的方式不同,可以明白为啥不能使用二级指针直接访问二维数组,同理三级指针寻址3次,四级指针4次,不管多少维数组都是和一维数组一样一次,所以如果我把断点继续指行就会应为访问未知地址挂掉。

再次将论点移到内存,我们所作的一切都只是对内存的访问,而C/C++一不小心就会出现对内存的非法的访问,所以了解内存是一件非常重要的事情,而指针只是访问操作内存的一种灵活的方式。和前面说的一样,任何指针占,所以任何一个4个字节的内存单元 一个int的大小 可以以任何的指针的方式访问内存,极度的灵活(如果对内存没有足够的理解同时就会导致极度的不安全),

到此这篇关于C语言数组和指针,内存之间的关系的文章就介绍到这了,更多相关数组和指针,内存之间的关系内容请搜索猪先飞以前的文章或继续浏览下面的相关文章希望大家以后多多支持猪先飞!

原文出处:https://blog.csdn.net/taolaodawho/article/details/122733588

[!--infotagslink--]

相关文章

  • C语言实现放烟花的程序

    这篇文章主要为大家详细介绍了C语言实现放烟花的程序,有音乐播放,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-02-23
  • C语言中的字符(char)详细讲解

    本篇文章主要介绍C语言中char的知识,并附有代码实例,以便大家在学习的时候更好的理解,有需要的可以看一下...2020-04-25
  • php中eval()函数操作数组的方法

    在php中eval是一个函数并且不能直接禁用了,但eval函数又相当的危险了经常会出现一些问题了,今天我们就一起来看看eval函数对数组的操作 例子, <?php $data="array...2016-11-25
  • Python 图片转数组,二进制互转操作

    这篇文章主要介绍了Python 图片转数组,二进制互转操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-09
  • 详解如何将c语言文件打包成exe可执行程序

    这篇文章主要介绍了详解如何将c语言文件打包成exe可执行程序,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-02-25
  • php数组操作 键名比较 差集 交集赋值

    本文章提供在量的数据中级操作实例有如对键名比较计算数组的差集 计算差集 给指定数组中插入一个元素 反转数组 交集赋值新的数组实例。 //定义回调函数 funct...2016-11-25
  • C#二维数组基本用法实例

    这篇文章主要介绍了C#二维数组基本用法,以实例形式分析了C#中二维数组的定义、初始化、遍历及打印等用法,具有一定参考借鉴价值,需要的朋友可以参考下...2020-06-25
  • C#数组的常用操作方法小结

    Array数组在C#中同样是最基本的数据结构,下面为大家C#数组的常用操作方法小结,皆为细小的代码段,欢迎收看收藏...2020-06-25
  • php curl模拟post请求和提交多维数组的示例代码

    下面一段代码给大家介绍php curl模拟post请求的示例代码,具体代码如下: <&#63;php$uri = "http://www.cnblogs.com/test.php";//这里换成自己的服务器的地址// 参数数组$data = array ( 'name' => 'tanteng'// 'passwor...2015-11-24
  • 浅谈redis key值内存消耗以及性能影响

    这篇文章主要介绍了浅谈redis key值内存消耗以及性能影响,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-02-07
  • C# 拷贝数组的几种方法(总结)

    下面小编就为大家带来一篇C# 拷贝数组的几种方法(总结)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2020-06-25
  • 详解分析MySQL8.0的内存消耗

    这篇文章主要介绍了详解分析MySQL8.0的内存消耗,帮助大家更好的理解和学习使用MySQL,感兴趣的朋友可以了解下...2021-03-23
  • PHP 二维数组根据某个字段排序的具体实现

    本文记录的要实现的功能类似于 MySQL 中的 ORDER BY,上个项目中有遇到这样的一个需求。 要求:从两个不同的表中获取各自的4条数据,然后整合(array_merge)成一个数组,再根据数据的创建时间降序排序取前4条。 遇到这个...2014-06-07
  • C语言中free函数的使用详解

    free函数是释放之前某一次malloc函数申请的空间,而且只是释放空间,并不改变指针的值。下面我们就来详细探讨下...2020-04-25
  • C#实现字符串转换成字节数组的简单实现方法

    这篇文章主要介绍了C#实现字符串转换成字节数组的简单实现方法,仅一行代码即可搞定,非常简单实用,需要的朋友可以参考下...2020-06-25
  • C语言中计算正弦的相关函数总结

    这篇文章主要介绍了C语言中计算正弦的相关函数总结,包括正弦和双曲线正弦以及反正弦的函数,需要的朋友可以参考下...2020-04-25
  • 详解C语言中的rename()函数和remove()函数的使用方法

    这篇文章主要介绍了详解C语言中的rename()函数和remove()函数的使用方法,是C语言入门学习中的基础知识,需要的朋友可以参考下...2020-04-25
  • c#将字节数组转成易读的字符串的实现

    这篇文章主要介绍了c#将字节数组转成易读的字符串的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-06-25
  • C#读取文件所有行到数组的方法

    这篇文章主要介绍了C#读取文件所有行到数组的方法,涉及C#针对文件及数组的相关操作技巧,需要的朋友可以参考下...2020-06-25
  • 将二维数组转为一维数组的2种方法

    如何将下面的二维数组转为一维数组。复制代码 代码如下:$msg = array(  array(    'id'=>'45',    'name'=>'jack'  ),  array(    'id'=>'34',    'name'=>'mary'  ),  array(    'id...2014-05-31