3. 指针与数组的比较
不同点:
数组:要么在惊天存储区域被创建(如全局数组),要么在栈上被创建。数组名对应着(而不是指向)一块内存,其地址与容量在生命周期内保持不变,只有数组的内容可以改变。
指针:可以随时指向任意类型的内存块,它的特征是“可变”,所以我们常用指针来操作动态内存。
两者特征比较的例子:
(1) 修改内容:
char a[] = “hello”;
a[0] = ‘x’ ; // ok
char *p = “hello”; //注意p指向常量字符串
p[0] = ‘x’; //错误。编译器不能发现该错误
指针p指向常量字符串”hello”(位于静态存储区,内容为hello),常量字符串的内容是不可以被改变的。企图修改常量字符串的内容而导致运行错误。
(2) 内容复制与比较
不能对数组名进行直接复制与比较,数组间复制应该使用标准函数strcpy进行复制。同理,比较b和a的内容是否相同,不能使用if( b == a )来判断,应该使用标准库函数strcmp进行比较。
语句p = a并不能将数组a的内容复制给指针p,而是把a的地址赋给了p,要想复制a的内容,可以先用库函数malloc为p申请一块容量为strlen(a) + 1 个字符的内存,再用strcpy进行字符串复制。同理,if( p == a) 比较的不是内容而是地址,应该用库函数strcmp比较。
// 数组
char a[] = “hello”;
char b[10];
strcpy( b, a); //不能用if( b == a)
…….
// 指针
int len = strlen(a);
char *p = (char *)malloc( sizeof(char) * ( len + 1) );
strcpy( p, a); // 不能用 p = a;
if( strcmp( p, a ) == 0 ) // 不能用if( p == a )
…….
(3) 计算内存容量
用运算符 sizeof 可以计算出数组的容量(字节数)。
下例中,sizeof (a)的值是12(注意别忘记了’’)。指针p指向a,但是sizeof(p)的值却是4(32位计算机),这是因为sizeof(p)得到的是一个指针变量的字节数,相当于sizeof(char *),而不是p所指的内存容量。
注意:当数组作为函数的参数进行传递时,改数组退化为同类型的指针。
Char a[] = “hello”;
Char *p = a;
cout << sizeof(a) << endl; // 12字节
cout << strlen(a) << endl; // 11字节
cout << sizeof(p) << endl; //4字节
cout << strlen(p) << endl; // 11字节
void Func(char a[100]) // 数组退化为指针
{
cout << sizeof(a) << endl; // 4字节
}