一个变量有一个地址,一个数组包含若干元素,每个数组元素都在内存中占用存储单元,它们都有相应的地址。所谓数组的指针是指数组的起始地址,数组元素的指针是数组元素的地址。
1、指向数组元素的指针(掌握)
一个数组是由一个连续的内存单元组成的。
数组名就是这块连续内存单元的首地址。一个数组也是由各个数组元素(下标变量)组成的。每个数组元素按其类型不同占有几个连续的内存单元。一个数组元素的首地址也是指它所占有的几个内存单元的首地址。
定义一个指向数组元素的指针变量的方法,与以前介绍的指针变量相同。
例如:
int a[10]; /*定义a为包含10个整数数据的数组*/
int *p; /*定义p为指向整型变量的指针*/
应当注意,因为数组为int 型,所以指针变量也应为指向int 型的指针变量。下面是对指针变量赋值:
p = &a[0];
把a[0]元素的地址赋给指针变量p。也即是:p指向a数组的第0号元素。
C语言规定,数组名代表数组的首地址,也就是第0号元素的地址。因此,下面两个语句等价:
p = &a[0];
p = a;
在定义指针变量时可以赋给初值:
int *p = &a[0];
它等效于:
int *p;
p = &a[0];
也可以写成:
int *p = a;
p,a,&a[0]均指向同一单元,它们是数组a的首地址,也是0号元素a[0]的首地址。应该说明的是p是变量,而a,&a[0]都是常量。在编程时需要注意。
数组指针变量说明的一般形式为:
类型说明符 *指针变量名;
其中类型说明符表示所指数组的类型。从一般形式可以看出指向数组的指针变量和指向普通变量的指针变量的说明是相同的。
2、通过指针引用数组元素(掌握)
C语言规定:如果指针变量p已指向数组中的一个元素,则p+1指向同一个数组中的下一个元素。
引入指针变量后,就可以用两种方法来访问数组元素了。
如果p的初值为p = &a[0];
则:
- p+i和a+i就是a[i]的地址,或者说它们指向a数组的第i个元素。
- *(p+1)或*(a+i)就是p+i或a+i所指向的数组元素,即a[i]。例如,*(p+5)或*(a+5)就是a[5]。
- 指向数组的指针变量也可以带下标,如p[i]与*(p+1)等价。
根据以上叙述,引用一个数组元素可以用:
- 下标法,即用a[i]形式访问数组元素。
- 指针法,即采用*(a+i)或*(p+i)形式,用间接访问的方法来访问数组元素,其中a是数组名,p是指向数组的指针变量,其处值p=a。
例10.9 输出数组中的全部元素。(下标法)
#include "stdio.h"
int main()
{
int a[10],i;
for(i = 0;i < 10;i++)
{
a[i] = i;
}
for(i = 0;i < 5;i++)
{
printf("a[%d]=%d\n",i,a[i]);
}
return 0;
}
运行结果为:
例10.10 输出数组中的全部元素。(通过数组名计算元素的地址,找出元素的值)
#include "stdio.h"
int main()
{
int a[10],i;
for(i = 0;i < 10;i++)
{
*(a+i) = i;
}
for(i = 0;i < 10;i++)
{
printf("a[%d]=%d\n",i,*(a+i));
}
return 0;
}
运行结果为:
例10.11 输出数组中的全部元素。(用指针变量指向元素)
#include "stdio.h"
int main()
{
int a[10],i,*p;
p = a;
for(i = 0;i < 10;i++)
{
*(p+i) = i;
}
for(i = 0;i < 10;i++)
{
printf("a[%d]=%d\n",i,*(p+i));
}
return 0;
}
运行结果为:
例10.12
#include "stdio.h"
int main()
{
int a[10],i,*p = a;
for(i = 0;i < 10;)
{
*p = i;
printf("a[%d]=%d\n",i++,*p++);
}
return 0;
}
运行结果为:
需要注意的问题:
- 指针变量可以实现本身的值的改变。如p++是合法的,而a++是错误的。因为a是数组名,它是数组的首地址,是常量。
- 要注意指针变量的当前值。
请看下面的程序。
例10.13 找出错误
#include "stdio.h"
int main()
{
int *p,i,a[10];
p = a;
for(i = 0;i < 10;i++)
{
*p++ = i;
printf("a[%d]=%d\n",i,*p++);
}
return 0;
}
运行结果:
显示报错。原因在于在for循环中执行一个循环条件*p++会执行两次
for(i = 0;i < 10;i++)
{
*p++ = i;
printf("a[%d]=%d\n",i,*p++);
}第一次执行时
i =0;
*p++ = 0;// a[0] = 0
p此时指向下一个地址p = &a[1];
在执行printf("a[%d]=%d\n",i,*p++);时等价于a[0] = a[1];然后p = &a[2];因为*p++是先用后加,把a[1]的值赋给a[0]后,指针又指向了a[2]的地址。执行十次后数组肯定溢出,并且除了第一次的a[0]元素是我自己赋的值外,其他值都是随机值。所以编译器报错啦!
当我把printf里的*p++改成*p时为什么会出现下图右边的结果呢?让我们来一起看一下叭!
for(i = 0;i < 10;i++)
{
*p++ = i;
printf("a[%d]=%d\n",i,*p);
}第一次执行时
i =0;
*p++ = 0;// a[0] = 0
p此时指向下一个地址p = &a[1];
在执行printf("a[%d]=%d\n",i,*p);时等价于a[0] = a[1];
第二次执行时
i =1;
*p++ = 1;// a[1] = 1
p此时指向下一个地址p = &a[2];
在执行printf("a[%d]=%d\n",i,*p);时等价于a[1] = a[2];
第3次执行时
i =2;
*p++ = 2;// a[2] = 2
p此时指向下一个地址p = &a[3];
在执行printf("a[%d]=%d\n",i,*p);时等价于a[2] = a[3];
。。。。。。
第十次执行时
i =9;
*p++ = 9;// a[9] = 9
p此时指向下一个地址p = &a[10];
在执行printf("a[%d]=%d\n",i,*p);时等价于a[9] = a[10];
多次执行这段代码发现结果只在这俩个结果循环显示。我本以为会每次执行这段代码a[]数组都是随机值,但结果并不是!具体是代码的问题,还是这个在线编译给这个数组分配的地址值是固定的我也不清楚,希望得到大佬们的解答,感谢!
最后听从朋友的建议改成以下代码可正常运行,原来for里最后一个条件也是可以写多个的,感谢朋友,又学到一个小知识。嘿嘿
#include "stdio.h"
int main()
{
int *p,i,a[10];
p = a;
for(i = 0;i < 10;i++,*p++)
{
*p = i;
printf("a[%d]=%d\n",i,*p);
}
return 0;
}
右边为输出结果
例10.14 改正(此为教学的改正方式)
#include "stdio.h"
int main()
{
int *p,i,a[10];
p = a;
for(i = 0;i < 10;i++)
{
*p++ = i;
}
p = a;
for(i = 0;i < 10;i++)
{
printf("a[%d]=%d\n",i,*p++);
}
return 0;
}
运行结果:
- 从上例可以看出,虽然定义数组时指定它包含10个元素,但指针变量可以指到数组以后的内存单元,系统并不认为非法。
- *p++,由于++和*同优先级,结合方向自右向左,等价于*(p++)
- *(p++)与*(++p)作用不同。若p的初值为a,则*(p++)等价a[0],*(++p)等价a[1]。
- (*p)++表示p所指向的元素值加1。
- 如果p当前指向a数组中的第i个元素,则
- *(p--)相当于a[i--];
- *(++p)相当于a[i++];
- *(--p)相当于a[--i];
3、数组名作函数参数(掌握)
4、指向多维数组的指针和指针变量(掌握)
由于时间关系,以上没有输出的我明日再写,码字不易,请给予多多鼓励!
如果你觉得对你有帮助,不要忘了关注点赞收藏哦,你的点赞收藏+关注将会给我源源不断的动力!让我们一起努力学习吧!成为学习工作和生活的高手!加油!!!晚安先,洗澡睡觉了俺