c语言指针例题解答

首先char*c[]为一个指针数组,对吧,这个相信大家都能看懂,千万不要把指针数组和数组指针搞混乱了。

首先先给大家讲解下什么是指针数组,什么又是数组指针呢?

eg:char *c[5] 因为[]的优先级高于*,所以c先与[]结合形成一个数组,类型为char* 类型,大小为5。里面存放属于char类型的指针。其实数组指针和二维指针有一定的相似之处,如果感兴趣,大家可以百度什么是二维指针,在以后的篇幅里,我也会详细给大家解释什么是二维指针,他们的联系是什么,区别又是什么呢?

而对于char(*c)[5] 因为()的优先级最高,所以c先和*形成一个指针,然后再与数组结合,这就形成了数组指针,即为指向数组的指针。它指向包含5个char类型元素的一维数组,此时c的增量以它所指向的一维数组长度为单位;

好了 弄懂了数组指针和指针数组,那么接下来继续解答该道题。

看这句代码,char**cp[]={c+3,c+2,c+1,c};

仔细看看,这里的**cp[]比上面的*c多了一个*。不知道大家看到这里有没有点疑惑呢?

多一个*号那他又代表什么呢?我来给大家举一个简单的例子吧。

char a='a';

char *b=&a;

上面这个代码的很简单吧!先定义了一个char变量,再由b指向a的地址。也就是说b里面存放的数据是变量a的地址。那么接下来就简单了,

再回到原题,char**cp[]也比char*c[]多一个*,那么不用说,cp[]里面也是存放着c[]的地址吧。看看cp[]里面的元素,c+3,c+2...果然是地址吧。

这里大家又有疑问了,为什么c+3是一个地址?那你们听说过 “数组名可看做一个地址”这句话吧!,那就对了,虽然c[]是个指针数组,那他仍然是个数组啊!!这就不难理解c+3,c+2等地址了吧。这里大家谨记可以吧数组名看做指针,但是数组名与指针又不是完全等价的!这个以后慢慢详解。那么接下来回到原题,看第三句,char***cpp=cp;

这句话我想很好理解吧!他是把一个存放指针数组地址的数组显式转化为一个指针。emmmmm,是不是有点头晕,转化来转化去的。看看他有3个*号,你就知道他是个三级指针了。

接下来看第一句printf, 参数是**++cpp;首先大家要知道输出三级指针指向的内容,他两个*又代表什么。

首先先看cpp前面有个++,什么意思呢?大家看看第三句代码,cpp指向cp,那么++cpp是不是就是cp[1]的地址啊,肯定是的。然后带个*号,取到cp[1]的值,而cp[1]的值是c+2,同时c+2又是一个地址,请大家好像体会这句话,给一个指针变量前面加一个*号就是表示他指针指向地址的值,那么这就很好理解**++cpp了。先是++cpp获取到cp[1]的地址,要获取cp[1]指向的内容再给前面加个*号就可以了,即*++cpp 就代表cp[1],即c+2。而c+2也是一个地址,要获取到c+2这个地址指向的内容,那再给前面加个*号就可以了。即为**++cpp;

这样**++cpp就代表指针数组c的第三个元素了,是一个字符串“world”,%s接收到这个字符串会把它都输出出来。即第一个printf打印WORLD.

举个简单的例子,给你们提个问题。

int a[5]={1,2,3,4,5}

int *p=a;

p++;//1

a++;//2

这两个哪个正确呢? 

1是正确的2是错误的。

要取到里面的元素  *(p),*(p+1)......;注意看这里也带个*,这个代码的意义是把数组显式转化为一个指针!,再结合图片的第二行和第三行,你现在应该清楚他的作用了吧。对,就是将数组显式转化为一个同维度的指针。。。好累明天继续更博。


 接下来回到问题,因为经过第一句**++cpp完后,cpp已经指向了cp[1],再次的*++cpp同样的道理,获取到cp[2]的元素,即c+1这个地址。然后再--,即为c这个地址,再带个*号,即为c这个地址下代表的值,其实也可以这么说,可以把指针数组看成一个二维数组。可以写成这样c[5][4]。那么c此时代表的值应该是二维数组的第一行吧。即c[0]。即此时为“hello”,别忘记了,后面还有个+3呢。因为hello本身也是个地址,如果大家再此又困惑,我给大家举了一个简单的例子。希望大家能认真看下帮助理解。


%s这个格式控制符需要接收一个字符串地址,c[0]地址获取到了,他就会把c[0]输出,这里给大家提个问题。他获取到c[0]地址后,会不会也把c[1]也输出出来呢?

哈哈  ,不要困惑了。并不会的。因为c[0]这个字符串“hello\0”,他会有个默认的'\0'结束符,结束%s的获取。我为什么会问这样的问题。因为要是同样的问题 遇到了二维数组大家想想又是什么样的情况呢?


大家好好对比看下。针对二维数组,让%s获取到c[0]即二维数组的第一行 时,他会把整个二维数组输出出来。直到遇到c[2][2]是个默认的‘\0’结束输出。 

通过一个例题 我举出这么多例子。。。。真累。不知道大家能不能理解呢?

接下来回到问题的第三个输出,参数为*cpp[-2]+3。不难理解。可以转化下,切记接下来这个让我受益终身的一句语法糖,*(a+1)等价a[1]。当你被指针搞得头大恨不得疯掉的时候,一定要想起这句语法糖。我不扯了。回到原题 。根据语法糖法则,把*cpp[-2]+3转化成**(cpp-2)+3。我想通过我这么详细的分析。这道题对大家不再有难度了吧。算了 ,送佛送到西。cpp经过上面两句已经指向了cp[2]这个地址,那么cpp-2指向cp[0]的地址没疑惑吧。然后带个*,即为*(cpp-2)即为cp[0]这个元素,即代表c+3这个地址,然后再带个*号为**(cpp+3),成为c[3]代表的元素。c+3这个地址代表的字符串是“SAYHI”,再+3,即把该字符串的地址再往后移动三位。这样就输出了“HI”。切记*(地址)=该地址指向的值。这句话可以套用。即就是这里面的地址可以是地址的地址。请各位好好理解吧。eg:*(*(地址的地址))


接下来第四句 cpp[-1][-1]大声告诉我,是什么!!!算了 ,我继续解答吧,谁让我就是这么这么好呢。老规矩,先转化成*(*(cpp-1)-1)+1;你们说下cpp此时指向谁呢?指向cp[1],如果你觉得我说错了,请好好回头仔细看看。很多人困惑了,为什么第三句输出不是移动了cpp了啊 。请你再仔细看看第一句,第二句和第三句printf里面的区别。对的,第一句第二句++cpp和第三句cpp-2是完全不相同的道理。就比如int a=1;a++和a+2能一样吗?显然是不一样的 a++后a自增为2。a+2执行完后a还是1。好了接下来回到原题。指向cp[1]后带个*号取到cp[1]的元素即为c+2又是个地址,在把这个地址--,得到c+1的地址。再带*号,得到c+1地址指向的值。为“New”,得到这个字符串地址后,再+1,输出EW。

此题结束 。答案选哪个呢?

我再问大家一个问题还是原来的那些数组,删掉那4句printf,重新写一句printf(“%c”,cpp[1][1][1])输出什么呢? 我给个答案 :输出A。

printf(“%c”,cpp[1][-1][1])又输出多少呢?答案:输出E。

请大家好好想想吧!指针的道路就是这样一步一步探索吧。我相信各位勇于尝试最终都能成为C指针高手。

 我才疏学浅,如若在我的博客里发现错误,请私信或评论,我会积极修改的。我知道CSDN博客里藏龙卧虎,还望大家不吝赐教。第一次写博客,乱七八糟的,希望大家多多宽恕。

猜你喜欢

转载自blog.csdn.net/huijiaaa1/article/details/78268562