【C语言进阶剖析】14.C语言中的单引号和双引号
一、单引号和双引号
-
C语言中的单引号用来表示字符字面量 C语言中的双引号用来表示字符串字面量
a表示字符字面量,在内存中占1个字节,a+1表示a的ASCII码加1,结果为b
"a"表示字符串字面量,在内存中占2个字节,"a"+1表示指针运算,结果指向"a"结束符
下面看一段单引号和双引号本质的代码:
#include <stdio.h> int main() { char* p1 = 1 ; char* p2 = 1; char* p3 = "1"; printf("%s, %s, %s", p1, p2, p3); printf( ); printf(" "); return 0; }
编译后会有警告,继续运行会发生段错误,如下:
究竟问题出现在哪里呢?下面先把打印语句注释掉并运行:
#include <stdio.h> int main() { char* p1 = 1 ; char* p2 = 1; char* p3 = "1"; //printf("%s, %s, %s", p1, p2, p3); //printf( ); //printf(" "); return 0; }
编译运行虽然有警告,但是没有段错误,那么段错误肯定是打印语句那里了
把打印语句去掉注释
#include <stdio.h> int main() { char* p1 = 1 ; char* p2 = 1; char* p3 = "1"; printf("%s, %s, %s", p1, p2, p3); //printf( ); //printf(" "); return 0; }
编译运行后果然出现了段错误
究竟是怎么回事呢?接着往下看
二、小贴士
-
字符字面量被编译为对应的 ASCII 码 字符串字面量被编译为对应的内存地址 printf 的第一个参数被当成字符串内存地址 内存的低地址空间不能在程序中随意访问
三、程序实例分析1
注意:字符 1 所对应的 ASCII 码就是十六进制的 0x00000031
0x08048000 这个内存地址是非常特别的内存地址,所有低于这个地址的内存空间是不能随便访问的,一旦访问,就会发生段错误。程序里面的字符串编译之后得到的地址都是大于 0x08048000,所以在程序中可以访问。
所以说 1 和 1 的内存地址都不能访问,访问了就会出现段错误。如果把 printf("%s, %s, %s", p1, p2, p3); 注释打印下面的语句会发生什么?
#include <stdio.h> int main() { char* p1 = 1 ; char* p2 = 1; char* p3 = "1"; //printf("%s, %s, %s", p1, p2, p3); printf( ); printf(" "); return 0; }
输出结果如下,还是会发生段错误:
四、程序实例分析2
字符字面量还是处于地址值区间,所以访问这个地址就会发生段错误。
如果把 printf( ); 注释掉
#include <stdio.h> int main() { char* p1 = 1 ; char* p2 = 1; char* p3 = "1"; //printf("%s, %s, %s", p1, p2, p3); //printf( ); printf(" "); return 0; }
编译就不会出错,而且换了一行:
五、容易混淆的代码
这段代码的本意是如果字符 c 是 " " 或者 " " 或者 " ",,则输入字符。
#include <stdio.h> int main() { char c = " "; while( (c == " ") || (c == " ") || (c == " ") ) { scanf("%c", &c); } return 0; }
编译运行后,发现程序并没有让我们输入:
在进行解释之前,先看下面这样会发生什么:
分析: 1)编译后字符串"string"的内存地址被赋值给变量 c
2)内存地址占用 4个字节,而变量 c 只占用1个字节
3)由于类型不同,赋值后产生截断
所以说 char c = " "; 赋值后产生截断,那么 while 里面的语句也就不会执行了,这段程序是混淆了字符和字符串的概念。
可以这么改:
#include <stdio.h> int main() { char c = ; while( (c == ) || (c == ) || (c == ) ) { scanf("%c", &c); } return 0; }
这样就能正确了:
六、小结
-
单引号括起来的单个字符代表整数 双引号括起来的字符代表字符指针 编译器接受字符和字符串的比较,无任何意义 编译器允许字符串对字符变量赋值,只能得到错误