在C语言中,有时候指针和数组等效,有的时候却不然。
什么时候不同?
1. 数组名本身就代表一个地址。指针代表地址的地址。所以,指针、数组的定义和可能有的多处的声明要严格匹配!不能将在一个地方定义的数组,在他处声明为指针。反过来也不行!(详见C专家p84)的
2. 指针和数组都可以在他们的定义中用字符串常量进行初始化。尽管看上去一样,但底层的实现机制却不同。定义指针时,编译器不为指针所指向的对象分配空间,它只是分配指针本身的空间。除非在定义的同时赋给指针一个字符串常量进行初始化。例如:char *p = "breadrain"; 注意,只有字符串常量才如此,不要指望为浮点数之类的常量分配空间:float *pip=3.14; 此语句错误,无法通过编译。(在ANSI C中,初始化指针时创建的字符串常量被定义为只读。如果试图通过指针修改这个字符串程序会出现未定义的行为)。数组也可以用字符串常量初始化,并且可以修改:char a[] = “breadrain“; 字符指针和字符数组,分别可以通过指针名、数组名直接打印输出。
什么时候相同?
1. 表达式中的数组名被编译器当做一个指向该数组第一个元素的指针。
2. 在函数参数的声明中,数组名被编译器当做指向该数组第一个元素的指针。
你要记住,在表达式中,指针和数组是可以互换的,因为它们在编译器里的最终形式都是指针,并且都可以进行取下标操作。因为数组的下标操作是建立在指针的基础上的。所有在函数调用时候传递给被调用函数的数组,在编译的时都被编译器改写为指针,即使被调用函数的型参声明为数组。当想把数组定义为函数的参数时,可以把它定义为数组,也可以定义为指针。不管选择那种方法,在函数的内部事实上获得的是一个指针!
课后练习(如果你预测的结果正确,说明你真正理解了C语言中的数组和指针):
2
3 char ga[] = "abcdefghi";
4
5 void my_array_func(char ca[10])
6 {
7 printf("sizeof array param = %#x \n", sizeof ca);
8 printf("addr of array param = %#x \n", &ca);
9 printf("addr of (ca[0]) = %#x \n", &(ca[0]));
10 printf("addr of (ca[1]) = %#x \n", &(ca[1]));
11 printf("++ca = %#x \n\n", ++ca);
12 }
13
14 void my_pointer_func(char *pa)
15 {
16 printf("sizeof ptr param = %#x \n", sizeof pa);
17 printf("addr of ptr param = %#x \n", &pa);
18 printf("addr of (pa[0]) = %#x \n", &(pa[0]));
19 printf("addr of (pa[1]) = %#x \n", &(pa[1]));
20 printf("++pa = %#x \n\n", ++pa);
21 }
22
23 int main()
24 {
25 printf("sizeof global array = %#x \n", sizeof ga);
26 printf("addr of global array = %#x \n", &ga);
27 printf("addr of (ga[0]) = %#x \n", &(ga[0]));
28 printf("addr of (ga[1]) = %#x \n\n", &(ga[1]));
29 my_array_func(ga);
30 my_pointer_func(ga);
31 }
输出:
addr of global array = 0x804a018
addr of (ga[0]) = 0x804a018
addr of (ga[1]) = 0x804a019
sizeof array param = 0x4
addr of array param = 0xbfd4ea50
addr of (ca[0]) = 0x804a018
addr of (ca[1]) = 0x804a019
++ca = 0x804a019
sizeof ptr param = 0x4
addr of ptr param = 0xbfd4ea50
addr of (pa[0]) = 0x804a018
addr of (pa[1]) = 0x804a019
++pa = 0x804a019
大功告成~~