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

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

  • 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语言数组和指针,内存之间的关系的文章就介绍到这了,更多相关数组和指针,内存之间的关系内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C语言创建链表错误之通过指针参数申请动态内存实例分析

    本文实例讲述了C语言创建链表中经典错误的通过指针参数申请动态内存,分享给大家供大家参考之用.具体实例如下: #include <stdio.h> #include <stdlib.h>// 用malloc要包含这个头文件 typedef struct node { int data; struct node* next;// 这个地方注意结构体变量的定义规则 } Node; void createLinklist(Node* pHder, int length) { int i =

  • 详解C语言-二级指针三种内存模型

    二级指针相对于一级指针,显得更难,难在于指针和数组的混合,定义不同类型的二级指针,在使用的时候有着很大的区别 第一种内存模型char *arr[] 若有如下定义 char *arr[] = {"abc", "def", "ghi"}; 这种模型为二级指针的第一种内存模型,在理解的时候应该这样理解:定义了一个指针数组(char * []),数组的每个元素都是一个地址. 在使用的时候,若要使用中间量操作元素,那么此时中间量应该定义为 char *tm

  • c语言循环加数组实现汉诺塔问题

    目录 简介 递归的汉诺塔解法(c语言) 循环实现汉诺塔问题(c语言) 简介 汉诺塔问题是学数据结构与算法的时候会遇到的问题,相信来看本文的读者应该都对汉诺塔问题有基本的了解,理论上所有的递归都可以改成循环,常规的做法是借助堆栈,但是我一想好像用循环加数组也可以实现,于是就有了本文,实现声明,本文最后出来的算法效率不高的,比直接用递归实现还要差很多,追求算法效率的同学就不用看这个了.题目:假设有3个柱子,分别为A.B.C,A柱子上有数量为n个的空心圆盘,从上到下序号分别为1...n,要求把A柱子中

  • 从头学习C语言之二维数组

    目录 二维数组: 语法: 二维数组存放方式: 二维数组的访问: 二维数组的初始化: 示例: 总结 二维数组: 引入一个想法: 一个变量:相当于一个点. 一维数组(也就是数组):点汇聚成为一条直线. 二维数组:直线汇聚成一个平面. 三维数组:平面汇聚成立体图形. 语法: 类型 数组名[常量表达式][常量表达式] int a[6][6];//6*6,6行6列 char b[4][5];//4*5,4行5列 double c[6][3];//6*3,6行3列 二维数组存放方式: 二维数组的访问: 数组

  • 深入理解C语言指针及占据内存空间

    原文链接:https://www.cnblogs.com/l-hh/p/12288613.html 第一.了解内存空间 本文章文字有点多,会有点枯燥,配合图文一起看可以缓解枯燥,耐心阅读哦!!! 先了解内存地址,才更好的理解指针! 我们可以把内存想象为成一列很长很长的货运火车,有很多大小相同的车厢,而每个车厢正好相当于在内存中表示一个字节.这些车厢装着不同的货物,就像我们的内存要存着各式各样的数据. 多啰嗦一下 我们平时在电脑上能够听音乐.看视频和文章,其实看到的这些东西就是内存中每个"车厢&q

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

    首先论证一维数组和一级指针之前的关系,我们常常使用一级指针指针的方式访问一维数组,只有对内存的理解到位才能理解它们直接的关系. 1.数组名是数组的首地址 2.对数组名取地址得到的还是数组的首地址 3.数组的访问方式其实就是首地址+偏移的寻址访问 我们在程序中会定义很多变量,有基本类型和自定义类型在进行开发的时候我对内存的访问访问就是通过变量名赋值的方式读写内存但是如果你看到的直接变量的符号名你将不可能理解内存.每一种类型都有字节宽度,char 1字节 short 2字节 int 字节float

  • C语言数组和指针的问题一道非常值得深思的笔试题

    最近笔试就遇到下面这道题,谁都不敢说自己的C/C++能有多精通,当然,工作一久,很多老毛病也就容易犯了,所以说,理论是真的很重要的,下面这道题,说实话还是挺基础的,虽然当时笔试被我给猜对了,但还是要深究一下具体的转换细节. 如题: #include <stdio.h> int main(void) { char *str[] = {"ab","cd","ef","gh","ij","k

  • 举例理解C语言二维数组的指针指向问题

    之前对数组的概念一直没有理解透彻,只觉得数组名就是个常量指针而已,用法和基本的指针差不多.所以当我尝试用二级指针去访问二维数组时,就经常会出错.下面就是刚开始写的一个错误的程序: #include <stdio.h> int main() { int iArray[2][3] = {{1,2,3},{4,5,6}}; int **pArray = NULL; pArray = iArray; printf("array[0][0] = %d\n", pArray[0][0]

  • C语言全方位讲解指针与地址和数组函数堆空间的关系

    目录 一.一种特殊的变量-指针 二.深入理解指针与地址 三.指针与数组(上) 四.指针与数组(下) 五.指针与函数 六.指针与堆空间 七.指针专题经典问题剖析 一.一种特殊的变量-指针 指针是C语言中的变量 因为是变量,所以用于保存具体值 特殊之处,指针保存的值是内存中的地址 内存地址是什么? 内存是计算机中的存储部件,每个存储单元有固定唯一的编号 内存中存储单元的编号即内存地址 需要弄清楚的事实 程序中的一切元素都存在于内存中,因此,可通过内存地址访问程序元素. 内存示例 获取地址 C语言中通

  • C语言例题讲解指针与数组

    目录 1.概要复习 2.指针与数组笔试题 2.1一维数组 2.2字符数组 2.3字符串数组 2.4字符串指针 2.5二维数组 1.概要复习 本篇的内容主要围绕指针与数组.指针与字符串等之间的关系,以及进一步理解sizeof .strlen 的使用与意义. 数组是指具有相同类型元素的集合,字符串常量是一个指向在连续空间里存放的字符的首字符的地址的指针.我们会在下面理解数组与字符串数组的不同. sizeof 是一个操作符,是计算类型空间大小的.strlen 是针对字符串的库函数,用来求字符串的长度.

  • C语言中的数组和指针汇编代码分析实例

    今天看<程序员面试宝典>时偶然看到讲数组和指针的存取效率,闲着无聊,就自己写了段小代码,简单分析一下C语言背后的汇编,可能很多人只注重C语言,但在实际应用当中,当出现问题时,有时候还是通过分析汇编代码能够解决问题.本文只是为初学者,大牛可以飘过~ C源代码如下: 复制代码 代码如下: #include "stdafx.h" int main(int argc, char* argv[]) {        char a=1;        char c[] = "

  • 详解C语言中的函数、数组与指针

    1.函数:当程序很小的时候,我们可以使用一个main函数就能搞定,但当程序变大的时候,就超出了人的大脑承受范围,逻辑不清了,这时候就需要把一个大程序分成许多小的模块来组织,于是就出现了函数概念:   函数是C语言代码的基本组成部分,它是一个小的模块,整个程序由很多个功能独立的模块(函数)组成.这就是程序设计的基本分化方法: (1) 写一个函数的关键: 函数定义:函数的定义是这个函数的实现,函数定义中包含了函数体,函数体中的代码段决定了这个函数的功能: 函数声明:函数声明也称函数原型声明,函数的原

  • C语言 数组指针详解及示例代码

    数组(Array)是一系列具有相同类型的数据的集合,每一份数据叫做一个数组元素(Element).数组中的所有元素在内存中是连续排列的,整个数组占用的是一块内存.以int arr[] = { 99, 15, 100, 888, 252 };为例,该数组在内存中的分布如下图所示: 定义数组时,要给出数组名和数组长度,数组名可以认为是一个指针,它指向数组的第 0 个元素.在C语言中,我们将第 0 个元素的地址称为数组的首地址.以上面的数组为例,下图是 arr 的指向: 下面的例子演示了如何以指针的方

  • 详解C语言中的指针与数组的定义与使用

    指针的特点 他就是内存中的一个地址 指针本身运算 指针所指向的内容是可以操作的 操作系统是如何管理内存的 栈空间 4M~8m的大小 当进入函数的时候会进行压栈数据 堆空间 4g的大小 1g是操作系统 全局变量 内存映射 可以对内存的内容修改修改硬盘的内容 一般在数据库中经常使用 内存的分配与释放 c语言分配内存的方法 // malloc(需要分配的大小): 这里的分配的大小需要对齐的2的指数 void *mem = malloc(size); 释放内存 // 一般分配的内容都是在堆空间中的 //

  • C语言的数组与指针你了解吗

    目录 前言 一.数组的定义 二.数组空间的初始化 1. char数组赋值 2.char数组硬件开发规范 二.数组与指针 总结 前言 自学笔记,没有历史知识铺垫(省略百度部分)C语言数组的概念及使用 一.数组的定义 char a[n]; 注意:数组与指针非常相似 二者的区别: 数组为常量,约定禁止二次赋值,即数组不应该出现在=左侧,如:a="HelloWorld";数组在声明时,会申请一段连续的内存空间,指针不会数组元素为变量,标记可以修改指向任意内存(实际上会copy右侧变量/常量到左

随机推荐