Chinaunix首页 | 论坛 | 博客
  • 博客访问: 8173930
  • 博文数量: 159
  • 博客积分: 10424
  • 博客等级: 少将
  • 技术积分: 14615
  • 用 户 组: 普通用户
  • 注册时间: 2010-07-14 12:45
个人简介

啦啦啦~~~

文章分类
文章存档

2015年(5)

2014年(1)

2013年(5)

2012年(10)

2011年(116)

2010年(22)

分类: C/C++

2011-11-15 22:06:29

作者:gfree.wind@gmail.com
博客:blog.focus-linux.net     linuxfocus.blog.chinaunix.net

话说有这样的一段代码:
  1. #include <stdlib.h>
  2. #include <stdio.h>


  3. static void divide_by_two(int num)
  4. {
  5.     while (num) {
  6.         printf("%d\n", num);
  7.         num /= 2;
  8.     }
  9. }

  10. int main()
  11. {
  12.     int num;
  13.     scanf("%d", &num);

  14.     divide_by_two(num);

  15.     return 0;
  16. }
某天,一个刚毕业的朋友——没有贬低应届生的意思哦~~~,开始负责维护这段代码。他呢,看到num /= 2,想起课程上讲过,整数右移一位,就等于除于二,并且右移操作比除法运算要高效的多。于是将 num /=2 改为了 num = num>>1。

代码变为
  1. #include <stdlib.h>
  2. #include <stdio.h>


  3. static void divide_by_two(int num)
  4. {
  5.     while (num) {
  6.         printf("%d\n", num);
  7.         num = num>>1;
  8.     }
  9. }

  10. int main()
  11. {
  12.     int num;
  13.     scanf("%d", &num);

  14.     divide_by_two(num);

  15.     return 0;
  16. }
编译成功后,当然不能忘了测试:
  1. [xxx@xxx-vm-fc13 test]$ ./a.out
  2. 10
  3. 10
  4. 5
  5. 2
  6. 1
  7. [xxx@xxx-vm-fc13 test]$ ./a.out
  8. 3
  9. 3
  10. 1
这位朋友对于结果很满意,于是将改动提交到了服务器,下班回家~~~

结果第二天就有同事来找他,说他的程序陷入了死循环:
  1. [xxx@xxx-vm-fc13 test]$ ./a.out
  2. -5
  3. -5
  4. -2
  5. -1
  6. -1
  7. -1
  8. 。。。。。。
  9. -1
这位朋友刚参加工作,第一次提交改动,就造成了这样的结果,自然很紧张。不是右移一位就等于除以2吗?究竟是怎么回事呢?虽然暂时不知道答案,也只能将改动rollback回去。

那么到底是什么原因呢?没错,右移一位就等于除以2,但是这里需要加一个条件,这里指的是正数。而对于有符号整数,且其值为负数时,在C99标准中对于其右移操作的结果的规定是implementation-defined.

在Linux上的GCC实现中,有符号数的右移操作的实现为使用符号位作为补充位。因此-1的右移操作仍然为0xFFFFFFFF。这导致了死循环。

阅读(1059611) | 评论(9) | 转发(19) |
给主人留下些什么吧!~~

Flashboy20052011-12-19 10:27:30

学习了。

编译了下,发现两段的编译结果稍有不同
移位的编译结果:
0x0000000100000e15 <right_shift_one+37>:        mov    -0x4(%rbp),%eax
0x0000000100000e18 <right_shift_one+40>:        sar    %eax
0x0000000100000e1a <right_shift_one+42>:        mov    %eax,-0x4(%rbp)
0x0000000100000e1d <right_shift_one+45>:        mov    -0x4(%rbp),%eax<

Rainyzzj2011-11-24 21:44:18

受教了,补码高位补符号位!

ceo10002011-11-16 22:41:20

上海保洁公司         http://www.shwdbj.net

上海保洁                http://www.shwdbj.net

上海市保洁公司     http://www.shwdbj.net



上海普陀保洁公司  回复 | 举报

重返人生2011-11-16 16:18:55

-1的右移操作仍然为0xFFFFFFFF
教训啊!~~