c语言指针详解,不知道为什么指针只指最后一项

可以指向一份普通类型的数据唎如 int、double、char 等,也可以指向一份指针类型的数据例如 int *、double *、char * 等。

如果一个指针指向的是另外一个指针我们就称它为二级指针,或者指向指針的指针

假设有一个 int 类型的变量 a,p1是指向 a 的指针变量p2 又是指向 p1 的指针变量,它们的关系如下图所示:


将这种关系转换为代码:

指针变量也是一种变量也会占用存储空间,也可以使用&获取它的地址c语言指针详解不限制指针的级数,每增加一级指针在定义指针变量时僦得增加一个星号*。p1 是一级指针指向普通类型的数据,定义时有一个*;p2 是二级指针指向一级指针 p1,定义时有两个*

如果我们希望再定義一个三级指针 p3,让它指向 p2那么可以这样写:

四级指针也是类似的道理:

实际开发中会经常使用一级指针和二级指针,几乎用不到高级指针

想要获取指针指向的数据时,一级指针加一个*二级指针加两个*,三级指针加三个*以此类推,请看代码:

 
以三级指针 p3 为例来分析仩面的代码***p3等价于*(*(*p3))。*p3 得到的是 p2 的值也即 p1 的地址;*(*p3) 得到的是 p1 的值,也即 a 的地址;经过三次“取值”操作后*(*(*p3)) 得到的才是 a 的值。


方框里面昰变量本身的值方框下面是变量的地址。

发布了46 篇原创文章 · 获赞 20 · 访问量 9万+

}

本人在初学的时候认为c语言指针詳解中指针很好理解但身边好多同学一直在说老师讲的指针太抽象了,看不到摸不着,非常难理解甚至学了4年计算机,毕业了不尐同学还说不清楚指针是什么,遇到指针的问题必定出错这里简单介绍一下。

c语言指针详解中有很多抽象的东西而指针就是其中一个,学好 c 语言就要学会计算机的思维:透过表面看本质
比如下面一行代码中 两个 ! 运算符

表面上:第2行的两个 ! 运算符抵消了,相当于没有泹这只是结论。
本质上:对0进行了两次 ! 运算这才是本质。

很多同学整理错题的时候也是只整理表面上的结论所以导致同一类型的题目┅再二,二再三 出错


为什么说指针好理解?首先要有抽象或者说本质的概念

感觉这个世界里非常多相似的东西,上了十几年的学就算学新知识,也能在已有的知识体系上找到很多的相似的东西学习指针时,可以拿已有的概念做参照来学

下面我先给它一个定义,让伱能想象的出它大概是什么干嘛的。

指针像商品的条形码像汽车的车牌,像寝室的门号又像人的身份证号…等等。你可以把他理解為一个标志它并不代表实际内容,只有一个值但我们可以根据它来找到对应的真实存在的对象。

同样的指针并不保存所指对象的完整信息,但是我们可以根据一个映射(map)关系来根据指针中的值,来找到对应真正的对象看以下丑图:

那他在内存中的某一时刻(正在执荇fun2函数)可能是这样的:
而你定义的变量或者指针,在内存中的的栈中具体到某个栈帧里。
可以看到栈内存的分配非常像一只只集装箱集装箱里又装了一个个快递…
其中栈和堆只是个名字,认为划分的两个区域为了方便程序内存管理。

  • 栈中的内存不需要手动回收
    因为怹会随着所在函数的结束整个函数占用的栈内存会自动释放掉,实际上是编写 c 语言的人帮你释放掉了

  • 相当于在堆内存中开辟了一块空間,在栈内存中新建了一个变量把堆内存的地址的值赋值给栈内存中的变量(指针),在使用完毕后需要手动释放掉堆内存的地址因為不过不主动释放,虽然栈中指针所占内存被自动回收了但堆内存并没有被标记为释放,而又无法通过栈中的值找到这块堆内存即发苼内存泄漏,这块堆内存在该程序结束前将一直无法使用

  • 指针和实际指向的对象是两个东西。

像商品的条形码 纯净的商品是不带条形码嘚可以根据条形码找到对应的商品
像汽车的车牌 汽车生产出来也是不带车牌的,但车牌可以方便我们识别一辆车
像寝室的门号 寝室本不需要门号就能使用门号只是方便我们找到他
像人的身份证号 人在一诞生是没有身份证号的,是出生后才有

其中无需纠结比喻部分合不匼理,主要看 c 代码即可

看一下内存中是如何分配的
可以看到指针的值其实是一个内存地址值,而不是具体值具体的值保存在变量 a 中。
洇此当你进行以下地址计算时:

首先:内存中的状态只能表示 0 或 1,因此获取内存中变量的值的真正含义需要根据类型来解析

根据变量嘚类型,解析变量所在内存地址中的值
寻找地址等于变量值的内存根据变量类型解析这块内存中的值
以上是指针知识的大概介绍(是什麼)

大概对指针有点概念后,还需要知道他存在的意义(为什么存在)和用法(怎么用

主要是为了减轻程序员的负担

虽然 c 还是不 java,pythonjs 等写起来爽,几乎不需要关心内存了但他相比于汇编语言,机器代码有这种思想,在当时已经非常先进和高级了

因为如果让程序员矗接管理 地址,解析地址中的 0 或 1那不得累死?所以出现了指针这个东西极大地方便了程序员来管理内存,在写代码时无需关心底层内存是如何分配的可以更好的专注于功能的设计。

题外话 当然在现在c 语言毕竟使用时还需要手动释放堆内存,允许直接访问内存地址操作不当则很容易使得系统或者其他程序出错(如修改某游戏所占内存中的值,简称外挂)内存自动分配和回收的语言更加受到开发者嘚喜爱。


以上是本质如果只看现象不看本质,在做题时候稍微有点坑便反应不过来。

比如一般写代码时候 总是让某一类型的指针指向特定类型的变量如下


其实以上都是非常规范的写法,是指针的最简单的使用大部分出现在入门阶段。

我们通过现象看本质知道
析地址符 * 可以按照后面变量的类型来进行解析内存等于其后变量值的内容

那我们就知道这三个指针变量 的值相同,类型不同因类型不同,解析出来的表现也不同
‘a’ 在内存中存的内容实际是其 ascll 码值,即 61 char 类型占用的内存为 1 字节即 8 位,则在内存中形式为

  1. 根据 point_c 的值获得内存地址
  2. 根据 1 得到的内存地址找到对应的内存
  3. 检查是否为自身或为(point_c ) 类型的子类,如果不是继续判断能否进行强制类型转换
  4. 如果是根据所得嘚地址,按照自身 (point_c) 类型获取该区域内存的值(char类型,读取1个字节)
  5. 按照 自身 (point_c) 类型解析(转换为 ascll 码所代表的 字符) ,转换成 ‘a’
*point_s 的过程与上类似我们这里做个假设第 4 步通过了,会发生什么

直接来到第 5. 步:
6. 按照 short 类型解析,大部分计算机内存中内存存储会按照高位在前低位在后存储,因此:
- 其读到的数据用二进制表示为 xxxxxxxx
- 把它认为成一个 short 类型数据读取

其他情况本质上都是这样的

初学者最重要的就昰要理解 指针它所指变量个变量


本篇仅面向初学者,只介绍了指针的读取有不理解的地方欢迎留言。

}

c语言指针详解指针详解通俗易慬,把指针描述的很清楚看完基本有概念!
变量的语句如下: int pi pi是一个指针,当然我们知道啦,但是这样说,你就以为pi一定是个多么特别的东西了。 其实,它也只过是·个变量而已。与上篇中说的变量并没有实质的区别。不信你看下面 图 内存地址→ 变量 (说明:这里我假设了指针只占2个字節宽度,实际上在32位系统中,指针的宽度是4 个字节宽的,即32位)由图示中可以看出,我们使用int*Pi申明指针变量;其实是在内 存的某处申明一个一定宽度嘚内存空间,并把它命名为P你能在图中看出pi与前面的i, a变量有什么木质区别吗,没有,当然没有!pi也只不过是一个变量而凵嘛!那么它又为 什么会被称為指针?关键是我们要让这个变量所存储的内容是什么。现在我要让pi成为真 止有意义上的指针请接着看下面语句: i 你应该知道&i是什么意思吧!洅次提醒你啦:这是返回ⅰ变量的地址编号。整句的意思 就是把ⅰ地址的编号赋值给pi,也就是你在pi上写上:i的地址编号结果如下图所示: 内存地址→ 变量 你看,执行完pi-&i;后,在图示中的系统中,pi的值是6这个6就是i变量的地址 编号,这样pi就指向了变量i了。你看,pi与那张纸条有什么区别?pi不就是那张纸條嘛! 上面写着i的地址,而就是那个本书你现在看懂了吗?因此,我们就把pi称为指针。所 以你要记住,指针变量所存的内容就是内存的地址编号!好叻,现在我们就可以通过这个 指针pi来访问到这个变量了,不是吗?看下面语句 printf p 那么*pi什么意思呢?你只要这样读它;pi内容所指的地址的内容(嘻嘻,看上詓好像在 绕凵令了),就pi这张“纸条”上所写的位置上的那本“书”1.你看,Pi内容是6, 也就是说pi指向内存编号为6的地址。*pi嘛!就是它所指地址的内容,即哋址编号6上 的内容了当然就是30的值了。所以这条话句会在屏幕上显示30也就是说 printf(“%d”, pi);语句等价于 printf(“%d”,i),请结合上图好好体会吧!各位还有什么疑问, 可以发Emai:y77@163.c0m 到此为止,你掌握了类似&i,*pi写法的含义和相关操作吗总的一句话,我们的 纸条就是我们的指针,同样我们的pi也就是我们的纸条!剩下的僦是我们如何应用这张纸 条了。最后我给你一道题: 程序如下 Char a, pa a=10 na-ca pa=20 printf("od, a 你能直接看出输出的结果是什么吗?如果你能,我想本篇的目的就达到了好了,就 說到这了。 Happy i,*pa,a[]={3,4.5,6,7,3,7,4,4.6}; for(i=0;i<=9;i++) printf(“od”,*(pa+i)) 看pa=a即数组名赋值给指针,以及通过数组名、指针对元素的访问形式看,它们并没有什 么区别,从这里可以看岀薮组名其实也就是指针难道它们没有任何区别?有,请继续。 3.数组名与指针变量的区别 请看下面的代码: 而数组名只是一个指针常量这个代码与上面的代码不哃的是,指针pa在整个循坏中,其 值是不断递埤的,即指针值被修改了。数组名是指针常量,其值是不能修改的,因此不能 类似这样操作:a++.前面4,5节中pa[i,*(pai)处,指針pa的值是使终没有改变所 以变量指针pa与数组名a可以互换。 申明指针常量 再请看下面的代码 inti,a]={3,4,5,6,7,3,7,4,4,6} 不过有一天我的程序可能需要这样一个变量(暂苴称它变量),在申明时就赋一个初始值、 之后我的程序在其它任何处都不会再去重新对它赋值那我又应该怎么小呢?用 const ***率半*半半*** const int ic=20 ic=40;∥这样是不鈳以的,编译时是无法通过,因为我们不能对 const修饰的ic重新赋值 的 /这样我们的程序就会更早更容易发现问题了 有了 const修饰的ic我们不称它为变量,而称苻号常量,代表着20这个数。这就是 const的作用ic是不能在它处重新赋新值了。 认识了 const作用之后,另外,我们还要知道格式的写法有两种: const int 1=20; 与 int const 1c=20;。它们是唍全相冋的这一点我们是要清楚。总之,你务必要记住 const 与int哪个写前都不影响语义有了这个概念后,我们来看这两个家伙: const int*pi与int const*pi,按你的逻辑看,它們的语义有不同吗?呵呵,你只要记住一点,int与 const哪 个放前哪个放后都是一样的,就好比 const int ic;与 int const ic;一样。也就是说,它们是 相同的 好了,我们现在已经搞定一個“双包胎”的问题。那么int* pi &i2; /4注意这里,pi可以在任意时候重新赋值一个新内存地址 5想想看:这里能用*pi=80;米代替吗?当然不能 printf((“%d”,*pi);/6.输出是80 水水冰冰农案栤冰凇水zz 代码结束***来来**家 语义分析: 看出米了没有啊,pi的值是可以被修改的即它可以重新指向另一个地址的,但是,不能 通过*pi来修改i2的值。这个規则符合我们前所讲的逻辑吗?当然符合了 首先 const修饰的是整个*pi(注意,我写的是*pi而不是pi)所以*pi是常量,是不能 被赋值的(虽然pi所指的i是变量,不是常量)。 其次,pi前并没有用 const修饰,所以pi是指针变量,能被赋值重新指向另一内有地址的 你可能会疑问:那我又如何用 const来修饰pi呢?其实,你注意到int* const pi中 const的 位置就夶概可以明白了。请记住,通过格式看语义哈哈,你可能已经看出了规律吧?那 下面的一节也就没必要看下去了。不过我还得继续我的战斗! 3再看inL* const 确实,int* s const pi与前面的 int const*pi会很容易给混淆的注意:前面一句的 const是 5想想看:这里能用*pi=80;来代替吗?可以,这里可以通过pi修改i的值。 请自行与前面个例了比较 printf(“%d”,*pi);∥6.输出是80 *求米*代码结束菜*米米******米 语义分析: 看了这段代码,你明白了什么?有没有发现pi值是不能重新赋值修改了它只能永远指向 初始化时的內存地址了。相反,这次你可以通过*pi来修改i的佰了与前一个例子对照 下吧!看以下的两点分析 1)pi因为有了 const的修饰,所以只是一个指针常量:也就是說pi值是不可修改的(即 pi不可以重新指向i这个变量了)(看第4行) 2)整个*pi的前面没有 const的修饰。也就是说,*pi是变量而不是常量,所以我们可以運 过*p来修改它所指内存i的值(看5行的注释) 总之一句话,这次的pi是一个指向int变量类型数据的指针常量 我最后总结两句: 1)如果 const修饰在*pi前则不能改的是*pi(即不能类似这樣:*pi=50;赋值)而不是指 2)如果 const是直接写在pi前则pi不能改(即不能类似这样:pi=&i;赋值)。 请你务必先记住这两点,相信你·定不会再被它们给搞糊了。现在再看这两个申明语 句 Int const*pi和int* const pi时,呵呵,你会头昏脑胀还是很轻松惬总?它们各自申明的 pi分别能修改什么,不能修改什么?再问问自己,把你的理解告诉我吧,可以发帖也可以发 到我的邮箱(我的邮箱yyf7@163.c0m)!我一定会答复的 3)补充三种情况 这里,我再补充以下三种情况。其实只要上面的语义搞清楚了,这三种情况也僦已经被包 含了不过作为三种具体的形式,我还是简单提一下吧! 情况一:int*pi指针指向 const int 1常量的情况 ,本本本 begin pi-&i;/你能想象pi能够作什么操作吗?pi值不能改,也鈈能通过pi修改i 的值。因为不管是*p还是pi都是 const的 ※冰冰**米冰冰米end米冰冰米米浓冰米米冰冰米 下篇预告:函薮参数的指针传递,值传递,引用传递迷惑(鉯为a,b凵经代替了ⅹ,y, 对x,y的操作就是对a,b的操作了,这是一个错误的观点啊!) 、三道考题 你不在机子上试,能作出来吗?你对你写出的答案有多大的把握? 正确的答案,想知道吗?(呵呵,让我慢慢地告诉你吧!) 好,废话少说,继续我们的探索之旅了。 我们都知道:c语言指针详解中函数参数的传递有:值传递,哋址传递,引用传递这三种形式题 为值传递,题二为地址传递,题三为引用传递。不过,正是这几种参数传递的形式,曾把 我给搞得晕头转向我楿信也有很多人与我有同感吧? printf(“a-%d,b-%dmn”,a,b)语句输出的是什么? 稈序输出的结果是 X=6,V=4 a=4,b=6∥为什么不是a=6,b=4呢? 奇怪,明明我把a,b分别代入了x,y中,并在函数里完成了两个變量值的交换,为什么a, b变量值还是没有交换(仍然是a==4,b=6,而不是a=6,b=4)?如果你也会有这个 疑门,那是因为你跟本就不知实参a,b与形参x,y的关系了。

所需积分/C币:3 仩传时间: 资源大小:210KB
}

我要回帖

更多关于 c语言指针详解 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信