// file1.c char str[] = "abcd"; // file2.c #include <stdio.h> extern char *str; int main() { str[0] = 'A'; printf("%s", str); }
编译正常,运行崩溃,为什么?
ringa_lee
char str[]跟char *str是有区别的。str[]是数组名是个标号,str虽然是数组的首地址,但是str[]本身不占用空间,只是一个符号。而char *str是要占用4个字节(64位为8个字节)空间的。
char str[]
char *str
假设在32位的机器上,链接器在链接的时候,将str[]开始的4个字节(即str[0],str[1],str[2],str[3])里面存储的值,copy给*str所在的4字节内存空间,这意味着*str占用的4个字节的内存空间的值变成了由"abcd"的的ascii值拼接起来组成的值。所以*str指向了一个未知的区域,对该未知区域读写都有可能引发segmentation fault。
*str
segmentation fault
如果你对*str执行printf("%d",str);的话这个值应当是(little endian)
97 + 98*256 + 99*256*256 + 100*256*256*256
补充一下:上面的阐述有个错误,*str其实并没有被分配4字节的存储空间,因为它是extern的。形象的说*str是"热脸贴冷屁股"贴到str[]上去的。错误产生的机理是一样的。
贴
extern char str[];指针不等于数组名。
char str[]
跟char *str
是有区别的。str[]是数组名是个标号,str虽然是数组的首地址,但是str[]本身不占用空间,只是一个符号。而char *str
是要占用4个字节(64位为8个字节)空间的。假设在32位的机器上,链接器在链接的时候,将str[]开始的4个字节(即str[0],str[1],str[2],str[3])里面存储的值,copy给
*str
所在的4字节内存空间,这意味着*str
占用的4个字节的内存空间的值变成了由"abcd"的的ascii值拼接起来组成的值。所以*str
指向了一个未知的区域,对该未知区域读写都有可能引发segmentation fault
。如果你对
*str
执行printf("%d",str);的话这个值应当是(little endian)补充一下:上面的阐述有个错误,
*str
其实并没有被分配4字节的存储空间,因为它是extern的。形象的说*str是"热脸贴冷屁股"贴
到str[]上去的。错误产生的机理是一样的。extern char str[];
指针不等于数组名。