最近在重新学习C语言,发现在学指针的时候,写到函数就会出现各种问题,百思不得其解,而且我也不太清楚指针和数组到底是什么关系。后来我查阅了相关的资料,觉得略有收获,现记录下来。
指针
对于C语言,大家知道和指针相关的运算有两种最基本的就是:* 取地址和& 取值,要明白对于
int *pa = a;
这样的pa是指针,申明了一个可以储存地址的空间,并申明这个地址对应的值的是int类型的。*pa是一个值(我觉得可以看成一个变量),而a为一个变量,如果后来初始化的话,一般这样写:
pa = &a;
这儿,pa的值就是a的地址,也就是取地址操作。*pa就可以取到pa所存的地址的实际数值,也就是a的值。
指针所储存的地址可以作加减运算,和普通的变量是一样的,没有什么区别。只是可以直接使用取值符得到其所存的地址的值,相对普通变量而言(指针变量也可以),可以做互逆的操作,就是取地址,取到相应的值所对应的地址。
说到这里基本上指针也就这些了。
数组名
对于数组而言,我们知道,数组是在一组连续的空间里储存的一些相关的值,申明数组可以用如下方式:
int a[5] = {1,2,3,4,5};
第一个值是a[0],这个0是指偏移量,是对数组开始的位置的偏移量,要确定具体值的位置就是 初始位置 + 偏移量 * 值所占空间大小 ,那么这个初始位置是什么呢,其实就是这个数组名。
- a 为数组名,看起来很像指针。
- &a 为指针,指的就是数组的位置。而且 &a = a
- &a[0] 也是指针,而且 &a[0] = &a = a
好,前面关于数组和指针的前提已经差不多阐述完毕了,接下来说说指针和数组名的关系,重点说说区别。
指针和数组名
首先我们有时候会这样使用指针:
int a[5] = {1,2,3,4,5};
int *pa;
pa = a;
那么这里发生了什么呢,其实是这样的,首先声明了一个包含5个变量的连续空间,并对其赋了值,这5个变量的最开始的位置就是a这个数组名。接下来又声明了一个指针,这里只申请了一个可以储存一个地址的空间,然后这个地址的值为a,也就是a数组的初始位置,故各种pa++,*pa,pa--之类的就都可以使用了。
我们就可以看出数组名和指针的不同了:
- 数组名的值一个地址,为数组的开始位置,虽然看起来像指针,却不是。
- 指针是一个储存地址的一种变量,可以储存数组名。
这样我们就可以解释这样的程序为什么不能运行了:
int a[5] = {1,2,3,4,5};
while(*a)
printf("%d",*a++);
原因就在于指针和数组本身就是不一样的两种东西,这里的a是一个数组名,是不能++的。
还有这种情况:
file 1:
extern int *a;
printf("%d",a[0]);
file 2:
int a[5] = {1,2,3,4,5};
这样就会出现在file 1中定义a为指针,在第file 2中a被定义为数组,那么在执行printf时a被认为是指针,那么在寻找a[0]的地址时对其进行&,&a = a,然后取a[0],得到a[0]的值为0,被认为是零指针,崩溃。
差不多就先写到这里,剩下的就下次再写,主要是关于参数传递和指针问题的,希望这次能把和指针相关的问题研究的稍微深一点。