char buf[] = “hello word”;
char *p = “hello world”;
buf[2] = ‘e’; | 正确 变量修改 |
---|---|
p[2] = ‘e’; | 错误 常量修改 |
这是为什么呢?
有初始值的数组,数组的初始值放在rodata段里面,地址是固定的,需要绝对地址访问。(注:绝对地址又名物理地址,相对地址又名逻辑地址)
数组名(地址)对应开辟的空间上的数值存放在栈上,编译器会去访问rodata段上的初始值然后取来初始化局部变量,因此数组的值才能修改。
而指针指向的是rodata段,是不能被修改的。
由此还有个知识点就是,数组名是地址,是个常量不可被修改,因此 buf = “hello world”;
是不对的,所以需要注意拷贝:
strcpy(buf,”hello world”); strcpy存在内存泄露
strncpy(buf,”hello world”,10);
我们可以写一个代码,测试下:
#include <stdio.h>
int main(void)
{
char buf[] = "hello word";
char *p = "hello world";
printf("%p, %p, %p\n", buf, p, "hello world");
return 0;
}
运行结果:
可以看到最后两个的地址次才是相同的,字符数组,后面的初始值虽然也是位于rodata,但是字符数组的特殊性,相当于buf[0]=‘h’;buf[1]=‘e’;buf[2]=‘l’;…buf[10]=’\0’;是从rodata处取得值复制到栈地址上。