C/C++ —— signed 与 unsigned 有符号和无符号数及其移位

unsigned int a = 0;
unsigned int b = -1;				// b 为 0xffffffff
unsigned int c = a - 1;				// c 为 0xffffffff
  • 注意:printf 的 %d 会按有符号数进行输出;

    printf("%d\n", (unsigned int)(0)-1);
    		// -1
    cout <<  (unsigned int)(0)-1 << endl;
    		// 4294967295
    cout << hex << (unsigned int)(0)-1 << endl;
    		// ffffffff
    

0. 补码

  • 对于负数
    • 补码:反码+1;
    • 反码:补码-1,再取反;
  • 以 32 位有符号数负数的补码为例:
    • -1:0xffffffff

1. 有符号位的正零与负零

  • 32 位有符号数:-0 也即 0x80000000,表示32位有符号数最大的负数,也即 -2147483648(== 0x7fffffff + 1)

    int x = 0x80000000;
    cout <<  x << endl;
    	// -2147483648
    int y = 0x7fffffff;
    cout << y+1 << endl;
    	// -2147483648
    

    0x8000 0000 == 0x7fff ffff + 1

2. 有符号位无符号位的移位运算

有符号数无符号数的左移均是逻辑移位,不会考虑符号位。无符号数右移,最左补零,有符号右移,保持符号位。

  • 无符号位:

    • 左移:最右补零;
    • 右移:最左补零;
    unsigned int x = 0xffffffff;
    
    cout << hex <<  (x << 1) << endl;
    		// fffffffe
    cout << hex <<  (x >> 1) << endl;	
    		// 7fffffff
    
  • 有符号位:

    • 正数:
      • 左移:第二位成为新符号位,有可能变为负数;
      • 右移:左边补零,保持符号位;
    • 负数
      • 左移:第二位成为新符号位,有可能变为正数;
      • 右移:左边补一,保持符号位;
    int x = 0xa0000000;
    cout << hex << (x << 1) << "\t" << dec << (x<<1) << endl;
    		// 40000000	1073741824,有符号位左移,第二位0成为新的符号位,负数变为正数
    cout << hex << (x >> 1) << "\t" << dec << (x>>1) << endl;
    		// d0000000	-805306368,有符号位右移,左端补1,保持符号位不变;
    
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

五道口纳什

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值