C编程小问题

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],超出这个范围的值会产生溢
出。另外还要清楚的就是负数的补码怎么表示。弄明白了这两点,这个问题其实就很简单了

 

问题1:

int i = -20;

unsigned j = 10;

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

解答1:int和unsigned int运算时int会自动转成unsigned int
int和unsigned int都是4字节(32位情况下)存储,区别是int最高位是符号位,用来表示正负
负数用补码存储,-20存储为11111111111111111111111111101100,这个东西转为unsigned int后就是一个很大的数4294967276了,所以最后结果是4294967286

 

解答2:这要看你的i+j的输出格式的  如果是%u 输出的话那肯定是没有负数的。应该是一个很大的正数。但是以%d输出的话就会转成int 类型了  就会是-10了

 

问题2:

下面的代码有什么问题?
    unsigned i ;
    for (i=9;i>=0;i--)
    {
        printf("%u\n",i);
    }

解答:原因是  i>=0  始终成立!
因为 i 被定义为无符号的整数。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值