【重要】数组与指针不等价

1

前言

数组与指针是不等价的,如:

  • 数组名不可以改变,而指向数组的指针是可以改变的。

  • 字符串指针指向的字符串中的字符是不能改变的,而字符数组中的字符是可以改变的。

  • 求数组长度时,借用数组名可求得数组长度,而借用指针却得不到数组长度。

2

区别一

数组名的指向不可以改变,而指向数组的指针是可以改变的。

请看如下代码:

#include <stdio.h>

int main(void)
{
 int a[5] = {0, 1, 2, 3, 4}, *p = a;
 char i;
 
 // 数组遍历方式一
 for ( i = 0; i < 5; i++ )
 {
   printf("a[%d] = %d\n", i, *p++);
 }
 
 // 数组遍历方式二
 for ( i = 0; i < 5; i++ )
 {
   printf("a[%d] = %d\n", i, *a++);
 }
 
 return 0;
}

数组遍历方式一:使用指针遍历数组元素,*p++等价于*(p++),即指针指向的地址每次后移一个单位,然后再取地址上的值。这里的一个单位是sizeof(int)个字节,至于为什么是sizeof(int)个字节可以查阅我的总结C语言指针变量的运算

数组遍历方式二:使用数组名自增遍历数组元素,编译出错,错误如下:

error: value required as increment operand

因为数组名的指向是不可以改变的,使用自增运算符自增就会改变其指向,这是不对的,数组名只能指向数组的开头。但是可以改为如下遍历方式:

for ( i = 0; i < 5; i++ )
 {
   printf("a[%d] = %d\n", i, *(a+i));
 }

这可以正确遍历数组元素。因为*(a+i)与a[i]是等价的。

3

区别二

字符串指针指向的字符串中的字符是不能改变的,而字符数组中的字符是可以改变的。

请看如下代码:

//字符串定义方式一
char str[] = "happy";

//字符串定义方式二
char *str = "happy";

字符串定义方式一:字符串中的字符是可以改变的。如可以使用类似str[3]='q'这样的语句来改变其中的字符。原因就是:这种方式定义的字符串保存在全局数据区或栈区,是可读写的。

字符串定义方式二:字符串中的字符是不可以改变的。原因就是:这种方式定义的字符串保存在常量区,是不可修改的。

具体可查阅:char *str与char str[]的区别

PS:关于C内存内存的总结可查阅我的总结:通俗易懂的C语言内存总结

4

区别三

求数组长度时,借用数组名可求得数组长度,而借用指针却得不到数组长度。

请看如下代码:

#include <stdio.h>

int main(void)
{
 int a[] = {0, 1, 2, 3, 4}, *p = a;
 char len = 0;
 
 // 求数组长度方式一
 printf("方式一:len=%d\n",sizeof(a)/sizeof(int));
 
 // 求数组长度方式二
 printf("方式二:len=%d\n",sizeof(p)/sizeof(int));
 
 return 0;
}

运行结果

方式一:len=5

方式二:len=1

求数组长度方式一:借用数组名来求数组长度,可求得数组有5个元素,正确。

求数组长度方式二:借用指针求数组长度,求得长度为1,错误。原因是:

p只是一个指向int类型的指针,编译器不知道其指向的是一个整数还是指向一个数组。sizeof(p)求得的是p这个指针变量本身所占用的字节数,而不是整个数组占用的字节数。

以上就是指针与其指向的数组的三个典型的区别:(1)遍历数组时需要注意;(2)修改字符串中的字符时需要注意;(3)求数组长度时需要注意。

推 荐 阅 读

【常见】getchar()、getche()、getch()的区别?
【知识点】#define与typedef的区别?
【常用】static有几种用法?

【常用】extern怎么用?

【易错】char *str与char str[]的区别?

【提高效率】你应该用复合赋值语句?

【知识点】来看一看volatile关键字

【知识点】C语言变量默认的初值是?

【易错】C语言内存对齐问题

【实用】关于数组与指针的总结

【编程练习】进制转换问题

【每日一句】

塑造自己的过程很疼,但最终你能收获一个更好的自己。

(0)

相关推荐