c语言中的0、‘0’、‘\0’、“0”及NULL
ASCII表字符的表示方法:
①单引号表示法:例如 ‘A’
②十进制表示法:直接使用该字符在ASCII表中的序号(即十进制值),例如65
③转义字符表示:使用 ‘\index’,其中index为该字符在ASCII表中的(八进制或16进制)值,例如 ’\101’(八进制)和 ’\0x41’(16进制)。使用该方法时,\ 后面的字符值只能使用该字符在ASCII表中的八进制值和16进制值,且默认使用八进制;使用16进制需要在 \ 后面加0x。
上面的’A’、65、’\101’、’\0x41’都是表示的大写字母A,其在ASCII表中的序号是65,即十进制数65,对应的八进制数是101,对应的16进制数是41。
0、’0’、’\0’、”0”、NULL
了解了上面ASCII表字符的表示方法之后,便不难理解标题中的几项了。
①0:ASCII表的第1个字符(NULL字符,打印出来的结果是一个空格,无任何显示),对应的十进制数0,对应的16进制数是0x00。
②’0’:表示字符0(阿拉伯数字0),ASCII表第48个字符,对应的八进制数是60,对应的16进制数是30。
③’\0’:使用转义字符表示的NULL,相当于①中直接使用的0。最常见的用法是隐式地表示字符数组结束。
④”0”:不是单个的字符,而是一个字符串。实际上它是两个字符,即’0’和’\0’。使用数组来存储时,该数组的长度是2。
⑤NULL:同①
举例说明(Windows环境下的VS2019):
例一:
#include<stdio.h>
void main() {
char a = 0;
printf("a = %d \na = %c \n", a, a);
char b = '0';
printf("b = %d \nb = %c \n", b, b);
char c = '\x30';
printf("c = %d \nc = %c \n", c, c);
}
编译运行结果:
例二:
#include<stdio.h>
void main() {
char a[] = "0";
char b[] = "A";
printf(" sizeof(a) = %d\n", sizeof(a));
printf(" a[0] = %d\n a[0] = %c\n", a[0], a[0]);
printf(" a[1] = %d\n a[1] = %c\n", a[1], a[1]);
printf(" sizeof(b) = %d\n", sizeof(b));
printf(" b[0] = %d\n b[0] = %c\n", b[0], b[0]);
printf(" b[1] = %d\n b[1] = %c\n", b[1], b[1]);
printf("%s,%s1", a, b);
}
编译运行结果:
在内存中,数组a和数组b各占用两个字节,数组a的两个字节里分别存储了’0’和’\0’,数组b的两个字节里分别存储了’A’和’\0’。但是,例二中最后面打印数组a和b时只打印了两个数组的第一个字节,(打印b时为了显得更直观,紧接着后面打印了一个1)。这是因为printf函数在打印字符串时,在遇到’\0’(即NULL字符)时会停止打印(即只打印遇到的第一个NULL字符之前的字符)。
例三:
#include<stdio.h>
void main() {
char a[4] = "";
a[0] = 65; //等效于 a[0] = 'A'
a[1] = 0;
a[2] = 66;
a[3] = 0;
for (int i = 0; i < sizeof(a); i++)
{
printf("a[%d] = %d a[i] = %c\n", i,a[i],a[i]);
}
printf("a = %s", a);
}
编译运行结果:
除了printf函数,c标准库函数中还有很多函数在遇到NULL时i会停止操作,比如strlen;也有很多函数在遇到NULL时仍然会继续操作,比如memcpy函数。在使用这些函数时要特别注意NULL。
例四:
#include<stdio.h>
void main() {
char a[] = "abc";
char b[] = "ABCDEF";
printf("before memcpy, strlen of b = %d\n", strlen(b));//数组b长度为7,但strlen会忽略最后面的'\0'
memcpy(b, a, sizeof(a));
for (int i = 0; i < sizeof(b); i++)
{
printf("b[%d] = %d\n", i, b[i]);
}
printf("after memcpy, strlen of b = %d", strlen(b));//注意,这里是3而不是6,因为memcpy遇到'\0'时不会停止拷贝(a[3] = '\0')。它会将a完全拷贝到b中(包括'\0'),导致b[3] = '\0'。
}
编译运行结果: