VS2022
char arr1[] = "abc";//双引号引起来的叫字符串
char arr2[] = { 'a','b','c' };//单引号引起来的叫字符
#include<stdio.h>
int main()
{
char arr1[] = "abc";
char arr2[] = { 'a','b','c' };
printf("%s\n", arr1);
printf("%s\n", arr2);
return 0;
}
为什么arr1是正常输出,然而arr2就是乱码呢?
答:在C语言中,字符串是以字符数组的形式表示的,并以空字符'\0'结尾,即字符串字面量是自动包含 结尾的空字符,对于上面的两个数组定义
char arr1[] = "abc";
char arr2[] = { 'a','b','c' };
arr2[]是由三个字符组成的数组,没有包含结尾的空字符,因此,如果把它视为字符串处理
printf("%s\n",arr2);它将不会在c后面停止而是,而是会继续读取内存直到遇到某个随机的空字符或未定义行为。
arr1[]是一个包含四个字符的数组:'a','b','c','\0'(空字符),这样在使用
printf("%s\n",arr1)时会正确输出“abc”并在'c'之后遇到空字符停下来。
接下来看一下二者的区别是什么:
char arr1[] = "abc";
和 char arr2[] = { 'a', 'b', 'c' };
这两个声明在功能上是相似的,它们都定义了一个字符数组,并初始化了它。但是,它们之间确实存在一些细微的差别:
- 字符串终止符:
char arr1[] = "abc";
:这会自动在数组的末尾添加一个空字符(null terminator,即 '\0')。因此,arr1
的实际内容是{'a', 'b', 'c', '\0'}
。这使得arr1
可以作为一个C风格的字符串使用。char arr2[] = { 'a', 'b', 'c' };
:这个声明不会自动添加空字符。arr2
的内容仅仅是{'a', 'b', 'c'}
。因此,如果你试图将它作为C风格的字符串来处理(例如,使用printf("%s", arr2);
),那么你会遇到未定义的行为,因为printf
会继续查找空字符来确定字符串的结束位置。
- 数组大小:
- 对于
arr1
,编译器会自动计算数组的大小为4(包括空字符)。 - 对于
arr2
,如果没有明确指定数组的大小,编译器会假设它的大小为3,因为只初始化了3个字符。
- 对于
- 可读性和意图:
- 使用
char arr1[] = "abc";
清晰地表示你正在创建一个字符串。 - 使用
char arr2[] = { 'a', 'b', 'c' };
可能意味着你只想存储三个字符,而不是一个完整的字符串。
- 使用
- 修改内容:
- 你可以修改
arr1
和arr2
中的字符,但重要的是要记得arr1
实际上有一个额外的空字符,而arr2
则没有。
- 你可以修改
- 初始化方式:
- 使用双引号
"abc"
是初始化字符串的常用方法。 - 使用大括号
{ 'a', 'b', 'c' }
允许你逐个初始化数组的元素,这在某些情况下可能更为灵活,尤其是当你不想初始化所有元素或需要更复杂的初始化逻辑时。
- 使用双引号
接下来怎么在VS里面查看区别,
找到调试->逐过程
保证接下来有黄色箭头出现
然后选择调试->窗口->监视->任选一个监视编号
在下方会有如下效果显示
在蓝色方框里填上自己要查看的语句
找到自己要调试的地方点击绿色箭头
会发现下面有了新的变化
可以看到两者的数组大小确实不一样。
arr1是4,arr2是3。
顺带一提,如果我自己给它定义一个数组的大小,比如3呢
char arrr2[3]={'a','b','c'}
很遗憾,依然会有乱码
abc烫烫烫烫烫烫烫烫烫烫烫烫烫烫蘟bc
那么这是为什么呢?
在 char arr2[3] = { 'a', 'b', 'c' };
这个声明中,你创建了一个大小为3的字符数组,并初始化了前三个元素。由于数组的大小是3,它只能包含这三个字符,并不包含字符串终止符 '\0'
。这意味着 arr2
不是一个以空字符结尾的C风格字符串。
所以你应该改成
char arr2[4]={'a','b','c'}
好奇这就来了,凭什么数组为4,就一定填充的是结束符'\0'呢?
然而,在 char arr2[4] = { 'a', 'b', 'c' };
这个声明中,你创建了一个大小为4的字符数组,并初始化了前三个元素。由于数组有额外的空间,编译器会自动在数组的最后一个位置(即索引为3的位置)放置一个空字符 '\0'
,作为字符串的终止符。这是因为当你初始化一个字符数组时,如果没有为所有元素提供显式的初始化值,且初始化的元素少于数组的大小,编译器会自动用0(对于字符类型来说,就是空字符 '\0'
)来填充剩余的数组元素。
学习记录,如有错误,还请指正。