首先,上代码:
int main()
{
int a = 012;
int b = '\012';
int c = '\0123';
return 0;
}
问:变量a、b、c的值各为多少?
a的值:因为是 012, 前面加了个0,所以是以八进制的形式赋值给了变量a,所以a的值为十进制的10.
b的值:对于单引号或者双引号的里的反斜杠,如果其后面跟了三个八进制常数位(0~7),则将其视为一个八进制数。所以,‘\012'的意思就是把八进制数 012 对应的ASCLL码的字符赋值给变量b,所以,变量b的值也是10,跟a的值一样。
c的值:同变量b,但它的反斜杠后面跟了4个八进制常数位,编译器最多只会认3个,所以后面的3,就是成单独得了。所以,这句的意思,可以理解为:先把八进制数 012 对应的ASCLL码的字符赋值给变量c,然后再把字符’3‘赋值给变量c(字符’3‘会覆盖掉前面的那个字符,在vs2017中,最多允许覆盖4次)。所以,将变量c的值,以字符(%c)打印出来就是字符’3‘,但如果以十进制数打印出来,就不清楚了。
同理,
int main()
{
int a = 0x12;
int b = '\x12';
int c = '\x0123';
printf("%d,%d,%d\n", a, b, c);
return 0;
}
0x12表示将后面的数以十六进制的方式赋值给变量a,即18.
'\x12'表示将十六进制数12,对应的ASCLL码的字符赋值给变量b,用十进制打印的话,也是18。
最下面的’x0123',编译时,编译器解释说,“对字符太大”,说明里面的值过大,具体大多少,我也不清楚。但可以大概的能说,\x,后面,一般跟两个数就可以了。
再来个字符串里的:
int main()
{
char *arr = "\\31\123234\x32gui";
char *brr = "\\31\129234\x32gui";
//char *crr = "\\31\123234\x32fui";
printf("%d\n", strlen(arr));
printf("%d\n", strlen(brr));
//printf("%d\n", strlen(crr));
return 0;
}
strlen(arr)的值为:11. 算作字符的: \\ 3 1 \123 2 3 4 \x32 g u i。所以,为11。
strlen(brr)的值为:12. 算作字符的: \\ 3 1 \12 9 2 3 4 \x32 g u i。所以,为12。因为9不是八进制常数位,所以到了只有12被认为是八进制的。
下面的strlen(crr),因为后面的 \x32f,太大,编译器不给通过,所以, \x后面最好只跟两个十六进制常数位(0~f)。
再顺带说下转义字符,一般的:\\ \' \" \? \n \r \a 等,它们都算作一个字符。,
再顺带说下,用反斜杠表示的八进制,遇到两种情况会停止,一种是读了3位,另一种就是还没读到3位时,碰到了非八进制常数位。