数组和指针 与 rodata段的联系

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处取得值复制到栈地址上。


版权声明:本文为qq_40674996原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。