1、要深入理解char* p和char a[]的区别,我们先要把数组和指针搞明白。
数组,其实很好理解,在大多数的计算机语言里是这样定义的:多个类型相同的数据的有序集合。
指针,相对来说要复杂点,由于我之前有写过有关指针的文章,这里就不再重复了,有关指针文章链接如下:http://t.csdn.cn/fj3Ok
2、char* p是字符指针,意思是这个指针里地址指向的数据类型为char。
注意,不管指向的数据类型是哪种,指针的本身只能为整型,因为指针里保存的是地址。
下面用代码来实验一下,代码如下:
char* char_p="hello world";
printf("char_p地址=%p,char_p=%s\n",char_p,char_p);
编译后运行如下图(图1):
3、char char_a[] 是字符数组,数组中存放的元素是字符,使用时要注意数组的空间大小。
下面用代码来实验一下,代码如下:
char char_a[15]="hello world";
printf("char_a=%s\n",char_a);
printf("char_a地址=%p,char_a[0]地址=%p",char_a,&char_a[0]);
编译后运行如下图(图2):
从图2中可以看出数组名char_a其实是一个地址,而且和数组第一个下标的地址相同。
也就是char_a=&char_a[0]。
编译器在处理这句代码时会把char_a数组从第一个下标char_a[0]开始依次把hello world\0逐个填
入。
另外需要说明的是在C语言里没有真正意义上的字符串,可以通过字符数组来表示字符串,因为它的元素地址是连续的,这样我们可以很方便的对每个数组下标的字符进行读写操作。
下面我们用for循环来把数组里的字符都打印出来,代码如下:
char char_a[15]="hello world";
for(int i=0;i<strlen(char_a);i++)
{
printf("char_a[%d]=%c,地址=%p\n",i,char_a[i],&char_a[i]);
}
编译后运行如下图(图3):
还有一点比较重要:在C语言中字符串常量是内存中一段连续的char空间,以'\0'(数字0)结尾,所以字符串常量能直接赋值给char* p和 char a[]
4、接下来我们来比较字符指针和字符数组两者有什么区别。
从上面的实践中我们知道,字符指针名和字符数组名都是一个地址。
不同的是字符指针是变量,而数组名是地址常量,不可以修改,只能在定义的时候对其初始化赋值。 因此p=a 可以,a=p不可以。
上面我用for循环打印了数组中的每个元素,这很正常,因为是数组所以可以通过数组下标访问到每个元素,而且还可以修改数组中的每个元素。那么字符指针怎么访问单个字符呢?其实字符指针也可以像数组那样通过下标访问到每个元素。代码如下:
char* char_p="hello world";
printf("char_p地址=%p,char_p=%s\n",char_p,char_p);
for(int i=0;i<11;i++)
{
printf("char_p[%d]=%c\n",i,char_p[i]);
}
编译后运行如下图(图4):
新手的话肯定会感觉很奇怪,字符指针怎么和字符数组一样的用法呢?
是的,确实是这样的。在访问每个元素这个功能上两者可以混用。但是两者却有着根本的区别。
字符指针之所以可以像字符数组一样访问每个元素,是因为字符串常量的原因。字符串常量是保存在常量区,且空间是连续的,返回第一个字符的地址为字符串常量的地址。因为是在常量区,所以字符指针只能访问每个元素,而不能修改它们。
通过上面的分析后,这两者的区别可以这样总结:
字符指针是变量,可以修改它的地址,但不可以修改它的地址指向的值。因为此时地址就是字符串常量的地址。
字符数组则刚好相反。它是常量,它的地址是在地址常量里,不可以修改,但是它的里面每个值可以修改。因为在字符串常量初始化字符串数组时,字符串常量的地址并没有赋值给数组名,而是依次把字符串常量里的字符逐个赋值到字符数组里。
原理已经讲明白了,为了加强理解我们可以这样做:上面讲了字符指针不可以修改它的地址指向的值,是因为地址来自字符串常量的地址。那么如果字符指针的地址来自字符数组的地址呢?按照原理一定可以修改每个元素的值了。我们用代码来验证一下,代码如下:
char char_a[15]="hello world";
char* char_p=char_a;
for(int i=0;i<11;i++)
{
printf("char_p[%d]=%c\n",i,char_p[i]);
}
char_p[0]='a';
char_p[1]='a';
char_p[2]='a';
printf("修改后的效果\n");
for(int i=0;i<11;i++)
{
printf("char_p[%d]=%c\n",i,char_p[i]);
}
编译运行后如下图(图5):
果然和预期的效果一样。 至此字符指针和字符数组的区别算是讲清楚了。另外再补充一下:字符指针能不能修改每个元素的值和它本身没有关系。而是看它的地址来自哪里。