在 C 语言中,字符串常量是作为字符数组来存储的,而数组名在大多数情况下可以被看作是指向数组第一个元素的指针。因此,字符串常量也可以被看作是指向其第一个字符的指针。
字符串常量
当你写下如 "Hello, World!"
这样的字符串时,编译器会在内存中创建一个包含这些字符的数组,并在数组的末尾加上一个空字符 '\0'
来表示字符串的结束。这个数组是一个字符数组,但是字符串常量的名字可以被当作指向这个数组第一个元素的指针。
举例
char *str = "Hello, World!";
在这行代码中,str
是一个指向 char
类型的指针,它指向字符串常量 "Hello, World!"
的第一个字符 'H'
。由于字符串常量是不可修改的,所以这个指针指向的内存区域是只读的。如果你尝试修改 str
指向的内容,比如 *str = 'J'
,编译器可能会允许你这样做,但这会导致未定义的行为,因为你在尝试修改只读内存。
注意
虽然字符串常量可以被看作是指针,但是它们并不是完全等价的。字符串常量的地址是固定的,你不能像修改普通指针那样修改字符串常量的地址。例如,下面的操作是不允许的:
char *str = "Hello, World!";
str = "Bye, World!"; // 这是允许的,因为你修改的是指针本身,而不是字符串常量
char *str = "Hello, World!";
*str = "Bye, World!"; // 这是错误的,因为你不能修改字符串常量
在第一个例子中,我们改变了 str
指针指向的地址(即改变了指针的值),这是合法的。在第二个例子中,我们尝试修改 str
指向的内存内容(即修改了字符串常量),这是不允许的。
总结来说,字符串常量在 C 语言中是一个指针,指向一个包含字符串字符的数组的第一个元素。这个指针可以被用来读取字符串的内容,但不能用来修改字符串的内容。
完整代码
#include <stdio.h>
int main() {
char *str = "Hello, World!";//等价于 char str[] = "Hello, World!";
//str = "Bye, World!"; // 这是允许的,因为你修改的是指针本身,而不是字符串常量
//*str = "Bye, World!"; // 这是错误的,因为你不能修改字符串常量
/* %p 用于打印指针的值,而 (void*)str 是将 char* 类型的指针转换为 void* 类型,
这样做是为了确保指针以正确的格式打印出来。在某些编译器中,直接使用 %p 打印 char* 类型的指针可能会出现问题,
因为 %p 期望的是一个 void* 类型的指针。
*/
printf("字符串首地址:%p\n", (void *)str) ;
printf("字符串首地址:%p\n", str) ;
printf("字符串:%s\n", str) ;
//printf("结果%s\n", *str) ;
return 0;
}
运行结果:
字符串首地址:0000000000404000
字符串首地址:0000000000404000
字符串:Hello, World!