iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C语言中数组和指针,内存之间的关系是什么
  • 311
分享到

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

2023-06-29 01:06:14 311人浏览 独家记忆
摘要

小编给大家分享一下C语言中数组和指针,内存之间的关系是什么,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!首先论证一维数组和一级指针之前的关系,我们常常使用一级指针

小编给大家分享一下C语言数组和指针,内存之间的关系是什么,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!

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

  • 数组名是数组的首地址

  • 对数组名取地址得到的还是数组的首地址

  • 数组的访问方式其实就是首地址+偏移的寻址访问

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

int a=10;// 往某段内存地址位x的内存 写入4个字节的数据10intp =(int)20; //往某段内存地址位x1的内存 写入4个字节的数据20int** p =(int**)30;// 往某段内存地址位x2的内存 写入4个字节的数据30int 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+偏移的方式 访问读或者写该指针指向数组中的内容

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

从上面的结果我们可以得到,别看到一级指针操作一维数组 和数组直接访问问之前只多了一层指针变量的寻址,但是含义就完全的变了。
如有些书上说数组作为实参会退化为指针
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)函数里面可以创建某些对象然后然后改变被调函数的指针,释放的时候同理,这样可以减少开发者所作的工作

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

可以用一级指针指针引用一位数组,以前刚学习的时候我自认而然的使用二级指针引用二维数组,发现是不可以。
进程的内存就是线性的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*宽度的寻址方式
三维数组同理

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

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

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变量中

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

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

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

以上是“C语言中数组和指针,内存之间的关系是什么”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注编程网其他教程频道!

--结束END--

本文标题: C语言中数组和指针,内存之间的关系是什么

本文链接: https://www.lsjlt.com/news/321887.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • c语言怎么计算字符串长度
    在C语言中,可以使用内置的strlen()函数来计算字符串的长度。该函数位于string.h头文件中。以下是一个示例代码: #inc...
    99+
    2024-05-14
    c语言
  • c语言怎么判断三个数的大小
    可以通过比较三个数之间的大小关系来判断它们的大小。以下是一个示例代码: #include <stdio.h> int ...
    99+
    2024-05-14
    c语言
  • c++排序函数sort怎么使用
    在C++中,使用sort函数对容器中的元素进行排序,可以按照默认的升序排列,也可以自定义排序规则。 下面是一个简单的例子,对一个ve...
    99+
    2024-05-14
    c++
  • c++中string的compare函数怎么使用
    在C++中,可以使用string的compare函数来比较两个string对象的大小。该函数的原型如下: int compare(c...
    99+
    2024-05-14
    c++
  • c++中string的compare函数有什么用
    c++中string的compare函数用于比较两个字符串的大小关系。该函数返回一个整数值,表示两个字符串的比较结果。 具体来说,如...
    99+
    2024-05-14
    c++
  • c++中print函数的功能有哪些
    在C++中,通常使用std::cout来进行输出操作,而不是使用print函数。std::cout是C++标准库中的一个输出流对象,...
    99+
    2024-05-14
    c++
  • c语言数据结构怎么使用
    在C语言中,数据结构可以通过定义结构体来实现。以下是一些常见的数据结构的使用方法示例: 链表: #include <std...
    99+
    2024-05-14
    c语言
  • c++中print函数的用法是什么
    在C++中没有内置的print函数,但可以使用cout来输出内容到控制台。cout是C++标准库中的一个输出流对象,可以使用<...
    99+
    2024-05-14
    c++
  • c++中concept的用法是什么
    在C++20中,Concept是一种新的语言特性,用于定义类型要求和约束。Concept可以被用来约束函数模板、类模板和普通函数的参...
    99+
    2024-05-14
    c++
  • c++中concept的作用是什么
    在C++中,concept的作用是定义一种通用的约束,用于限制模板参数的类型范围。通过使用concept,可以在编译时对模板参数进行...
    99+
    2024-05-14
    c++
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作