C语言理论篇——带你轻松玩转指针与动态分配内存

2019.12.11

1.什么是指针以及它的作用

储存数据的内存是一串连续的字节单元格,如图

内存图解.png

每一个单元格都有自己的地址,而指针则是用来储存某个地址的变量
那么我们什么时候需要用到指针呢?下图作为引例
值改变.png

使用函数返回值可以实现一个值的改变,而return只能返回一个值,改变两个及以上的值便需要指针
地址详解.png

说明:首地址是指第一个字节的地址,其实指针并没有如图所示的指向作用,那样画方便理解,指针就是地址,通过它可以间接访问或修改所存的地址内的值,依图举个例子,可以将0x100(地址)理解为一个小区中的房间门牌号,1(所存的值)当做租客,而指针就是保安室(登记有所有门牌号即地址),通过保安室可以查到门牌号,再使用一个*即登门拜访
一个指针所占用空间:64位系统占8个,32位系统占4个

2.指针的类型

指针的运算指针+i是指指针所指地址向右移动i*sizeof(所指向的变量类型)
指针之间还可以作减法,仅限于数组内,作差的结果是两个地址间相差的元素个数
整型指针:int *a
字符指针:char *a
指针数组:int *a[5],一个大小为5,每个元素都是指针
数组指针:int (*a)[5] 只能指向二维数组,且二维数组中的一维数组长度为5
如int a[3][5];
 int (*p)[5]=a;
常量指针:指向地址的值不能改变
int a=20,b=30;
int const *p=&a;const(常量)在前
*p=&b;//可以改变指针的指向关系
指针常量:指向关系不能变,值可以变
int *const p1=&a;*指针在前
*p1=50;//可以改变里面的值
谁在前谁不变
二级指针: int**p;里面存的是一个一级指针的地址,指针本身是有地址的

二级指针图解.png

二级以上的指针如图类推
指针函数:int *add(); 先忽略*,add()是个函数,它的返回类型是int*指针(即地址)
函数指针:int (*p)(int ,int); 由于()的优先级更高,所以无法先忽略*,所以从*开始读,是一个指针,它指向的类型是一个函数,函数的返回类型是整型
函数名其实是一个指针常量,它指向该函数代码的首地址
int minus(a,b);
p=minus;
(*p)(a,b);等价于minus(a,b)

☆补充说明:相信很多初学指针的小伙伴都有这样的一个困惑,指针只能用地址来赋值,为什么常量字符串可以使用char* name="xiaoming",常量字符串为什么可以这样赋值,其实啊,真正赋值给name的还是一个地址双引号" "做了以下三件事:
1.在内存中开辟了一片连续的空间
2.将常量字符串放进去,并在末尾添加\0
3.最后返回那一片连续内存空间的首地址
所以常量字符串的赋值不需要使用malloc为字符指针分配内存空间,直接使用双引号即可,这样的表达是没有任何问题的,这是指针中容易搞混的地方

3.指针和数组的关系

1.指针与一维数组之间可以相互转换,数组名是数组的首地址,可以直接赋给指针

指针与数组的转换.png

被注释掉的输出结果一样
数组是指针常量,所以不能对首地址进行自增(++)或自减(--)运算,本质上是不能对首地址赋值如a=a+1而*(a+1)则是正确的,数组地址移动方式和指针是相同的,a+i其实是a+i*sizeof(数组类型),所以对首地址先进行移动,然后再显示新地址内的值
指针移动等同数组.png

所以其实a[i]等价于*(a+i)
2.数组指针:数组指针是二维的,(只能说)类似于二级指针,要访问里面的单个元素,需要使用**
先定义一个数组指针
int a[2][3]={{1,2,3},{4,5,6}};int (*p)[3]=a;
数组指针图解.png

而对p+1则是纵向的移动,移动到4所在地址
数组指针.png

上图则是运行效果
也可以换一种方式理解为什么需要两个**:前面提到一维数组等价于指针,所以二维数组就相当于一维数组里的元素全是指针,*p访问到第一个元素还是指针即地址,所以还需要一个*才能访问到真正的第一个元素

4.动态分配内存

常量区 常量 const int a=1;程序结束时释放
静态区 static 直到程序结束才会被释放内存
静态变量只会被初始化一次,生命周期从开始(预编译)到程序结束为止
栈 局部变量 int a=10;作用于代码块,出了代码块就释放
堆 自己申请的内存(malloc calloc realloc) 必须自己释放内存
为什么需要分配内存:
1.存储的数据需要延长生命周期
2.一个指针变量需要存储数据,变量本身只能存地址,不能存数据,需要分配内存空间来存储数据,必须为指针变量分配内存空间,用malloc,引入stdlib.h
如何分配内存
char *name;
name=(char *)malloc(10*sizeof(char));
格式:malloc(需要对应数据类型空间的数量*sizeof(对应的数据类型)),前面需要对其声明,(数据类型*)
动态分配内存的本质:malloc函数在内存中申请了一片连续指定大小的空间单元格,然后返回那片连续单元格的首地址
name=(char*)realloc(name,20*sizeofz(char))//重新分配已经分配的动态内存
格式:realloc(需要重新分配的指针名,对应数据类型空间的数量*sizeof(对应的数据类型))
为了不浪费空间,程序结束之间需要释放申请的内存,free(name);
以上具体的使用会在实战篇中体现

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,830评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,992评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,875评论 0 331
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,837评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,734评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,091评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,550评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,217评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,368评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,298评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,350评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,027评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,623评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,706评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,940评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,349评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,936评论 2 341

推荐阅读更多精彩内容