美文网首页
一道C++面试题和补码、无符号数减法运算

一道C++面试题和补码、无符号数减法运算

作者: Aspirinrin | 来源:发表于2017-11-24 17:02 被阅读455次

面试题在文章第4节。在看面试题之前,可以先看一下1-3节的知识点。

1. 补码

Two's Complement(二补数、补码)是对二进制数的数学运算,运算过程为:对二进制序列每一位取反(0->1; 1->0),再加1。

bits 取反 补码
011 100 101
010 101 110
111 000 001

2. 计算机中有符号数的表示

计算机中的数值类型分为整数型和浮点数型,有符号数在最高位设置符号位,其余低位均为数值位。数值位一律采用补码形式存储,并参与计算。采用补码的形式表示有符号数至少有两大好处。

  • 符号位和数值位统一参与运算,不用区分正、负,加法和减法实现简单;
  • 数据的原码和补码之间的相互转换不需要依赖额外硬件电路。

下面分别介绍有符号数的表示方法

2.1 整数

正整数

正整数的补码是其二进制表示,与原码相同。
例如,在整数类型占用4字节(32位)的系统中,+5的补码是00000000 00000000 00000000 00000101。最高位0表示该数值为正数,其余31位表示数值大小。

负整数

负整数的补码需要对其绝对值的二进制表示进行补码运算。
例如,-5的补码是11111111 11111111 11111111 11111011。最高位为1,表示该数值为负数,其余31位表示数值大小。

在进行运算时,CPU并不会区分是正数还是负数,而是直接进行计算,这正是前面介绍的符号位和数值位的统一。
例如,a=10, b=-5,则a+b的运算过程如下:

  00000000 00000000 00000000 00001010 (10)
+ 11111111 11111111 11111111 11111011 (-5)
===========================================
  00000000 00000000 00000000 00000101 (5)

如果,a=1, b=5,则a-b首先转换成加法a+(-b),再进行计算,过程如下:

a-b ==> a+(-b)
  00000000 00000000 00000000 00000001 (1)
+ 11111111 11111111 11111111 11110110 (-10)
===========================================
  11111111 11111111 11111111 11110111 (-9)   

对于正整数(最高位为1),将非符号位的二进制位直接转换成十进制,就表示该正数的实际大小。如果一个数是负整数,如何将其补码转换成十进制大小呢?补码运算即可。
例如上面的11111111 11111111 11111111 11110111,最高位符号位是1,所以该数为负数,补码运算之后为00000000 00000000 00000000 00001001,大小为9,所以表示-9

2.2 浮点数

pass

3. 为什么是补码?

为什么两个数相减a-b用补码形式a+(-b)进行计算的结果是正确的?不妨看一下对b进行补码的过程绝对值的二进制序列取反,再加1。取反在计算机的逻辑电路中就是开关的闭合状态取反即可,即1->0,0->1。如果用数学算式表达的话,对一个bit位b的取反运算可以写成

取反b = 1-b (*)
b=0时,取反b为1,1-b=1;
b=1时,取反b为0,1-b=0;
所以算是(*)可以表达取反运算

综上,a-b的计算过程可表达为(8bit为例)

a-b == a+(-b) == a+(11111111-b+1) == a+(b的补码形式)
在8bit系统中,11111111 + 1 == 00000000,溢出。
所以,a+(11111111-b+1) = a+(0-b) = a - b

可以看出,补码运算的实现效果巧妙地利用了因计算机系统位数限制而产生的溢出现象。

4. 一个C++面试题

下面代码打印多少?

#include <iostream>

int main(int argc, char **argv)
{
    std::cout << 25u - 50;
    return 0;
}

答案是4294967271

25uunsigned int类型,50为int类型。在这两种操作数进行-运算时,int被提升为unsigned int型,运算变为25u - 50u,结果也应该是unsigned int类型。经过对-50u进行补码运算后带入加法运算,-25的二进制表示形式被存入内存,即11111111 11111111 11111111 11100111(int为32位),在打印时按无符号数处理,则直接转换成十进制正整数为4294967271

11111111 11111111 11111111 11100111 =
2^31 + 2^30 + ... + 2^5 + 2^2 + 2^1 + 2^0 =
2^5(1-2^27) / (1-2) + 7 =
4294967271

更多面试题和答案:24 Essential C++ Interview Questions

相关文章

  • 一道C++面试题和补码、无符号数减法运算

    面试题在文章第4节。在看面试题之前,可以先看一下1-3节的知识点。 1. 补码 Two's Complement(...

  • Lecture 4

    2.2 定点加法、减法运算 2.2.1 补码加法 2.2.2 补码减法 2.2.3 溢出概念与检测方法 2.3 定...

  • 进制

    一.原码反码补码基本概念 1.数据在计算机内部是以补码的形式储存的 2.数据分为有符号数和无符号数 无...

  • 原码,反码和补码

    在计算机内,有符号数有3种表示法:原码、反码和补码,所有数据的运算都是采用补码进行的。 正数的原码,反码,补码都相...

  • 1.2.17_C++ ++ 和 -- 运算符重载

    C++ 重载运算符和重载函数 递增运算符( ++ )和递减运算符( -- )是 C++ 语言中两个重要的一元运算符...

  • 原码、反码、补码

    1.原码反码补码基本概念 数据在计算机内部是以补码的形式储存的 数据分为有符号数和无符号数(最高位为符号位)无符号...

  • java基础篇三(运算符号、表达式与语句)

    一、运算符 赋值运算符:= 一元运算符: +,正号-,负号!,非~:取补码,如下例子: 算数运算符: +,加法-,...

  • 1.2.15_C++ 关系运算符重载

    C++ 重载运算符和重载函数 C++ 语言支持各种关系运算符( < 、 > 、 <= 、 >= 、 == 等等),...

  • 4. Python运算符

    算数运算符 加法运算符: + 减法运算符: - 乘法运算符: * 除法运算符: / 幂运算符: ** 整除运算符:...

  • 1.2.16_C++ 输入/输出运算符重载

    C++ 重载运算符和重载函数 C++ 能够使用流提取运算符 >> 和流插入运算符 << 来输入和输出内置的数据类型...

网友评论

      本文标题:一道C++面试题和补码、无符号数减法运算

      本文链接:https://www.haomeiwen.com/subject/svefbxtx.html