C语言:sizeof

 
c语言深度剖析 (二) sizeof 关键字 你有可能死在他手上
2011-09-23 20:19

1.5,最冤枉的关键字----sizeof

1.5.1,常年被人误认为函数

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 前加 unsignedconst 等关键字但不能加 sizeof。好,记住:sizeof 

计算变量所占空间大小时,括号可以省略,而计算类型(模子)大小时不能省略。一般情况下,

咱也别偷这个懒,乖乖的写上括号,继续装作一个“函数”,做一个“披着函数皮的关键字”。

做我的关键字,让人家认为是函数去吧。

1.5.2sizeofint*p 表示什么意思?

sizeofint*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)  的值是多少?

}

1.4signedunsigned 关键字

我们知道计算机底层只认识 01.任何数据到了底层都会变计算转换成 01.那负数怎么

存储呢?肯定这个“-”号是无法存入内存的,怎么办?很好办,做个标记。把基本数据类

型的最高位腾出来,用来存符号,同时约定如下:最高位如果是 1,表明这个数是负数,其

值为除最高位以外的剩余位的值添上这个“-”号;如果最高位是 0,表明这个数是正数,

其值为除最高位以外的剩余位的值。

这样的话,一个 32 位的 signed   int 类型整数其值表示法范围为:-         -1位的

31 2 31 2

char 类型数其值表示的范围为-         -1。一个 32 位的 unsigned   int 类型整数其值表示法

7 2 7 2

范围为:0       -1位的 char 类型数其值表示的范围为 0   -1。同样我们的signed  

32 2 8 2

键字也很宽恒大量,你也可以完全当它不存在,编译器缺省默认情况下数据为 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 循环内,当的值为时,a[0]的值为-1。关键就是-1 在内存里面如何存储。

我们知道在计算机系统中,数值一律用补码来表示(存储)。主要原因是使用补码,可

以将符号位和其它位统一处理;同时,减法也可按加法来处理。另外,两个用补码表示的数

相加时,如果最高位(符号位)有进位,则进位被舍弃。正数的补码与其原码一致;负数的

补码:符号位为 1,其余位为该数绝对值的原码按位取反,然后整个数加 1

按照负数补码的规则,可以知道-1的补码为 0xff-2 的补码为 0xfe……当的值为 127

时,a[127]的值为-128,而-128  char 类型数据能表示的最小的负数。当继续增加,a[128]

的值肯定不能是-129。因为这时候发生了溢出,-129 需要位才能存储下来,而 char 类型

数据只有位,所以最高位被丢弃。剩下的位是原来位补码的低位的值,即 0x7f

继续增加到 255 的时候,-256 的补码的低位为 0。然后当增加到 256 时,-257 的补

码的低位全为 1,即低八位的补码为 0xff,如此又开始一轮新的循环……

按照上面的分析,a[0] a[254]里面的值都不为 0,而 a[255]的值为 0strlen 函数是计

算字符串长度的,并不包含字符串最后的‘\0’。而判断一个字符串是否结束的标志就是看

是否遇到‘\0’。如果遇到‘\0’,则认为本字符串结束。

分析到这里,strlen(a)的值为 255 应该完全能理解了。这个问题的关键就是要明白 char

类型默认情况下是有符号的,其表示的值的范围为[-128,127],超出这个范围的值会产生溢

出。另外还要清楚的就是负数的补码怎么表示。弄明白了这两点,这个问题其实就很简单了。

留三个问题:

1),按照我们上面的解释,那-0 +0 在内存里面分别怎么存储?

2),int i = -20;

unsigned  j = 10;

i+j 的值为多少?为什么?

3),下面的代码有什么问题?

unsigned i ;

for (i=9;i>=0;i--)

{

printf("%u\n",i);

}

btw ,nothingl3 :(4) int i=3, j=sizeof(i+++ ++i);

执行完上述语句后,i和j的值?

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值