C语言中的单引号用来表示字符字面量,C语言中的双引号用来表示字符串字面量。
'a'表示字符字面量,在内存中占一个字节
'a' + 1 表示'a'的ASCII码加1,结果为'b'
"a"表示字符串字面量,在内存中占两个字节
"a" + 1表示指针运算,结果指向"a"结束符'\0'
下面的程序合法吗?
char *p1 = 1;
char *p2 = '1';
char *p3 = "1";
示例:单引号和双引号的本质
#include <stdio.h>
int main()
{
char* p1 = 1 ;
char* p2 = '1';
char* p3 = "1";
printf("%s, %s, %s", p1, p2, p3); // 会发生segmentation fault,p1和p2处于内存的低地址内存空间
printf('\n'); // 会发生segmentation fault,访问内存的低地址空间
printf("\n");
return 0;
}
输出结果:
小贴士:
- 字符字面量被编译为对应的ASCII码
- 字符串字面量被编译为对应的内存地址
- printf的第一个参数被当成字符串内存地址
- 内存的低地址空间不能在程序中随意地访问
示例:混淆概念的代码
#include <stdio.h>
int main()
{
char c = " ";
while( (c == "\t") || (c == " ") || (c == "\n") )
{
scanf("%c", &c);
}
return 0;
}
输出结果:
看上去,while内部没有被执行,为什么?
下面的语句发生了什么?
char c = "string";
分析
- 编译后字符串”string”的内存地址被赋值给变量c
- 32bit机器上,内存地址占用4个字节,而变量c只占用1个字节
- 由于类型不同,赋值后产生截断
上述代码将双引号改成单引号即可正常工作。
总结
- 单引号括起来的单个字符代表整数
- 双引号括起来的字符代表字符指针
- C 编译器接受字符和字符串的比较,无任何意义
- C 编译器允许字符串对字符变量赋值,但只能得到错误