字符串常量的类型是 const char[],其本质是字符数组而非指针。1. 字符串字面量如 "abc" 被编译器处理为 const char[4],在表达式中会退化为 const char 指针;2. 不可将字符串常量赋值给 char,因其指向只读内存区域,修改会导致未定义行为;3. char arr[] = "hello" 将内容复制到栈上,char ptr = "hello" 则指向字符串常量区;4. arr 可修改,ptr 指向的内容不可修改;5. sizeof(arr) 返回数组大小,而 sizeof(ptr) 返回指针大小;6. 使用数组适用于需修改内容或独立拷贝的情况,使用指针适用于仅需读取且节省内存的情况;7. 避免用非 const 的 char 接收字符串常量;8. 相同字符串常量可能被合并存储,std::string 自动管理内部拷贝。

在C++中,字符串常量(比如
"Hello, world!")的类型其实是
const char[],也就是说它是一个字符数组。虽然很多新手会把它和
char*混为一谈,但它们并不完全等价,背后涉及存储方式、生命周期以及使用时的一些细节。

字符串常量的类型到底是啥?
当你写
"abc"这样的字符串字面量时,编译器会自动把它当作一个
const char[4]类型来处理(包括结尾的
\0)。这其实是个数组类型,而不是指针。不过,在大多数表达式上下文中,这个数组会“退化”成指针,也就是变成
const char*。这也是为什么你可以这样写:

const char* ptr = "hello";
但注意,你不可以这样写:
立即学习“C++免费学习笔记(深入)”;
char* ptr = "hello"; // 错误:不能把 const char* 赋给 char*
因为字符串常量是只读的,试图修改会导致未定义行为。

字符数组和指针在存储上的区别
这里我们重点看两种常见写法:
char arr[] = "hello"; char* ptr = "hello";
这两者的区别主要体现在内存布局和可变性上。
1. 存储位置不同
arr
是一个局部数组,它的内容会被复制到栈上。ptr
是一个指针,指向的是字符串常量区(通常位于只读内存区域)。
也就是说,
arr的值是独立的一份拷贝,而
ptr只是引用了已有的字符串。
2. 是否可以修改
-
arr
中的内容是可以修改的:arr[0] = 'H'; // 合法
-
ptr
指向的内容是不能改的:ptr[0] = 'H'; // 未定义行为,可能崩溃
3. sizeof 表现不同
sizeof(arr)
得到的是数组的实际大小,比如 6 字节(5个字母 + \0)。sizeof(ptr)
得到的是指针本身的大小,通常是 4 或 8 字节(取决于平台)。
使用建议:什么时候用数组?什么时候用指针?
-
如果你需要修改字符串内容,或者想确保每个变量都有自己的副本,用字符数组更合适:
char name[] = "Tom"; name[0] = 'J'; // 成功变成 "Jom"
-
如果你只是读取字符串,不需要修改,而且希望节省内存或提高效率,那么用指针更好:
const char* greeting = "Welcome!";
*避免使用非 const 的 char 来接收字符串常量**,这是不安全的做法,C++标准已经明确反对。
一些容易忽略的细节
- 多个相同的字符串常量可能会被合并(称为“字符串池”优化),也就是说两个不同的指针可能指向同一个地址。
- 如果你用
std::string
,它内部会自动管理一个字符数组的拷贝,所以不会有上面这些问题。
基本上就这些。理解字符串常量的类型和字符数组与指针的区别,能帮你写出更安全、高效的 C++ 代码。









