引入
咱们先上代码,看看这个问题是怎么引出的~
下面分别通过字符串指针和字符数组定义了一个字符串hello world!
,然后我们通过遍历的形式把第二个字符e
改成o
,预期中的结果应该是两种方式都输出hollo world!
对吧?
#include <stdio.h>
int main() {
char str[] = "hello world!";
char *p_str = "hello world!";
for (int i = 0; i < 12; i++) {
if (i == 1) {
str[i] = 'o';
}
printf("%c", str[i]);
}
printf("\n");
for (int i = 0; i < 12; i++) {
if (i == 1) {
*p_str = 'o';
}
printf("%c", *p_str);
p_str++;
}
return 0;
}
输出结果:
hollo world!
h
Process finished with exit code -1073741819 (0xC0000005)
可以看到程序并没有如我们预想的那样,而是异常退出了,这是为什么呢?
解释
C语言中,使用指针定义的字符串一般称为字符串常量(也叫字符串字面量),他是存储在内存中的静态区(数据区)的;而使用字符数组定义的字符串是存储在内存中的栈区的。在C语言中,栈区和堆区是属于可以读写的部分,而静态区和代码区在内存中通常是只读区域,所以字符串指针指向的地址空间是不可变的,上面的代码我们尝试去操作修改静态区的内存,因此引发错误退出的情况。
总结
如果我们要定义一个可供修改的字符串,一定记得使用字符数组的形式哦,字符串指针定义的字符串属于不可变类型!