这篇文章记录我学习C语言关键字的相关知识,旨在记录一些进阶的知识。
目录
一、关键字sizeof
(一)、sizeof常被误认为函数
sizeof 是关键字不是函数,其实就算不知道它是否为 32 个关键字之一时,我们也可以借助编译器确定它的身份。看下面的例子:
int i=0; A),sizeof(int); B), sizeof(i); C), sizeof int; D), sizeof i;
毫无疑问, 32 位系统下 A), B)的值为 4。那 C)的呢? D)的呢?
在 32 位系统下,通过 Visual C++6.0 或任意一编译器调试,我们发现 D)的结果也为 4。这就奇怪了, sizeof 后面的括号呢?没有括号居然也行,那想想,函数名后面没有括号行吗?由此轻易得出 sizeof 绝非函数。
好,再看 C)。编译器怎么怎么提示出错呢?不是说 sizeof 是个关键字,其后面的括号可以没有么?那你想想 sizeof int 表示什么啊? int 前面加一个关键字?类型扩展?明显不正确,我们可以在 int 前加 unsigned, const 等关键字但不能加 sizeof。
好,这下记住了: sizeof 在计算变量所占空间大小时,括号可以省略,而计算类型(模子)大小时不能省略。一般情况下,咱也别偷这个懒,乖乖的写上括号,继续装作一个“函数”,做一个“披着函数皮的关键字”。做我的关键字,让人家认为是函数去吧。
(二)、sizeof(int)*p 表示什么意思?
sizeof(int) *p 表示什么意思?
留几个问题, 32 位系统下:
int *p = NULL;
sizeof(p)的值是多少?
sizeof(*p)呢?
int a[100];
sizeof (a) 的值是多少?
sizeof(a[100])呢? //请尤其注意本例。
sizeof(&a)呢?
sizeof(&a[0])呢?
int b[100];
void fun(int b[100])
{
sizeof(b);// sizeof (b) 的值是多少?
}
以上问题将会在讲解指针的部分给大家详细讲解,敬请关注,持续更新中……
二、singned、unsigned关键字
我们知道计算机底层只认识 0、 1.任何数据到了底层都会变计算转换成 0、 1.那负数怎么存储呢?肯定这个“-”号是无法存入内存的,怎么办?很好办,做个标记。把基本数据类型的最高位腾出来,用来存符号,同时约定如下:最高位如果是 1,表明这个数是负数,其值为除最高位以外的剩余位的值添上这个“-”号;如果最高位是 0,表明这个数是正数,其值为除最高位以外的剩余位的值。
这样的话,一个 32 位的 signed int 类型整数其值表示法范围为: - 2^31~ 2^31-1; 8 位的char 类型数其值表示的范围为- 2^7 ~ 2^7 -1。一个 32 位的 unsigned int 类型整数其值表示法范围为: 0~ 2^32 -1; 8 位的 char 类型数其值表示的范围为 0~ 2^8 -1。同样我们的 signed 关键字也很宽恒大量,你也可以完全当它不存在,编译器缺省默认情况下数据为 signed 类型的。
上面的解释很容易理解,下面就考虑一下这个问题:
int main()
{
char a[1000];
int i;
for(i=0; i<1000; i++)
{
a[i] = -1-i;
}
printf("%d",strlen(a));
return 0;
}
此题看上去真的很简单,但是却鲜有人答对。答案是 255。别惊讶,我们先分析分析。
for 循环内,当 i 的值为 0 时, a[0]的值为-1。关键就是-1 在内存里面如何存储。我们知道在计算机系统中,数值一律用补码来表示(存储)。主要原因是使用补码,可以将符号位和其它位统一处理;同时,减法也可按加法来处理。另外,两个用补码表示的数相加时,如果最高位(符号位)有进位,则进位被舍弃。正数的补码与其原码一致;负数的补码:符号位为 1,其余位为该数绝对值的原码按位取反,然后整个数加 1。
按照负数补码的规则,可以知道-1 的补码为 0xff, -2 的补码为 0xfe……当 i 的值为 127时, a[127]的值为-128,而-128 是 char 类型数据能表示的最小的负数。当 i 继续增加, a[128]的值肯定不能是-129。因为这时候发生了溢出, -129 需要 9 位才能存储下来,而 char 类型数据只有 8 位,所以最高位被丢弃。剩下的 8 位是原来 9 位补码的低 8 位的值,即 0x7f。当 i 继续增加到 255 的时候, -256 的补码的低 8 位为 0。然后当 i 增加到 256 时, -257 的补码的低 8 位全为 1,即低八位的补码为 0xff,如此又开始一轮新的循环……
按照上面的分析, a[0]到 a[254]里面的值都不为 0,而 a[255]的值为 0。 strlen 函数是计算字符串长度的,并不包含字符串最后的‘\0’ 。而判断一个字符串是否结束的标志就是看是否遇到‘\0’ 。如果遇到‘\0’ ,则认为本字符串结束。
分析到这里, strlen(a)的值为 255 应该完全能理解了。这个问题的关键就是要明白 char类型默认情况下是有符号的,其表示的值的范围为[-128,127],超出这个范围的值会产生溢出。 另外还要清楚的就是负数的补码怎么表示。 弄明白了这两点, 这个问题其实就很简单了。