基础野:细说有符号整数

简介:

Breif                              

 本来只打算理解JS中0.1 + 0.2 == 0.30000000000000004的原因,但发现自己对计算机的数字表示和运算十分陌生,于是只好恶补一下。

 本篇我们一起来探讨一下基础——有符号整数的表示方式和加减乘除运算。

 

Encode                              

  有符号整数可表示正整数、0和负整数值。其二进制编码方式包含 符号位 和 真值域。

  我们以8bit的存储空间为例,最左1bit为符号位,而其余7bit为真值域,因此可表示的数值范围是{-128,...,127},对应的二进制补码编码是{10000000,...,01111111}。

  从集合论的角度描述,我们可以将十进制表示的数值范围定义为集合A,将二进制表示的数值范围定义为集合B,他们之间的映射为f。f(a)=b,其中a属于A、b属于B。并且f为双射函数。因此有符号整数表示方式具有如下特点:

  1. 可表示的数值范围小;

  2. 十进制表示的数值范围与二进制表示的数值范围的元素是一一对应的,两者可精确映射转换。(相对浮点数而言,某些二进制表示的数值只能映射为十进制表示的数值的近似值而已);

  3. C语言中虽然没有规定必须采用补码来对有符号数进行编码,但大部分实现均是采用补码。而Java和C#则明确规定采用补码来表示有符号数。

 

Sign-extended                        

  符号扩展运算用于在保持数值不变、符号位不变的前提下,不同字长的整数之间的转换。

  例如现在我们要将8bit的10000100扩展为16bit,那么我们只要将高8bit设置为与原来符号位相同的值(这里是1)即得到1111111110000100,而其数值和符号并不产生变化。

 

Truncation                          

  截断会减少位数,并对原始值取模。模为2^n,n为截断后的位数。

  例如现在将16bit的100000100000000100截断为8bit,那么结果为00000100,而模是2^8。

 

Addition                            

  注意:位级运算均是模数运算,即加减乘除后均会对运算结果取模,并以取模后的结果作为终止返回。

  有符号整数加法的运算顺序:

  1. 算术加法(由于采用补码对有符号数进行编码,则是已经将负数转换为正数存储,所以含负数的加法只需要直接执行算术加法即可);

  2. 执行截断操作。

  示例1,两个4bit的有符号数相加(3+6):

  0011

+0110

  1001,然后执行截断得到1001,发生正溢出得到 -7

  示例2, 两个4bit的有符号数相加(-3+6):

        1101

      +0110

       10011,然后执行截断得到0011,发生正溢出得到 3

 

Subtraction                          

  有符号整数减法的运算顺序:

  1. 将减法转换为加法(对减数取补码);

  2. 算术加法;

  3. 执行截断操作。

  示例1,两个4bit的有符号数相减(-5-6):

 1011

-0110

对减数求补码后,减法转换为加法

  1011

+1010

 10101,然后执行截断得到0101,发生负溢出得到5

   示例2,两个4bit的有符号数相减(-5-(-6)):

 1011

-1010

对减数求补码后,减法转换为加法

  1011

+0110

 10001,然后执行截断得到0001,得到1

 

Multiplication                         

  对于乘法实质上就是通过移位操作和加、减法组合而成。

  1. 将乘数以二进制形式表示,并以连续的1作为分组。如-5的二进制形式为(1)0(11),从左至右可分成2组分别是(1)、(11)。

  2. 以n表示每组的最高位的指数,以m表示每组最低位的指数。如第一组n=m=3,第二组n=1而m=0。

  3. 根据公式(x<<n+1)-(x<<m)对每组进行运算,并将结果相加。如(假设被乘数为-1)

            第一组:(1111)<<(3+1) - (1111)<<3 = 0000 - 1000 = 0000 + 1000 = 1000

            第二组:(1111)<<(1+1) - (1111)<<0 = 1100 - 1111 = 1100 + 0001 = 1101

            两组相加:1000 + 1101 = 10101,截断得到0101,等于十进制数值5.

      2.4. 对结果取模。

 

Division                            

   对于除法实质上就是通过移位操作和加、减法组合而成,且根据除数是否为2的n次幂(n为正数)区别处理。

  1. 对于被除数为2的n次幂(n为正数)的情况,除法公式为:a>>n,如-6/4等价于6/(2^2),则可转换为移位操作-6>>2即可。然后再对结果取模。

  2. 对于被除数不为2的n次幂(n为正数)的情况,则情况复杂不少。运算步骤如下:(实质上我们就是按这个步骤做十进制除法的)

   2.1. 对负数取补,提取符号乘积。

      2.2. 高位对齐,在除数值小于被除数值的前提下,让除数的位数等于被除数;若执行高位对齐后,除数值大于被除数时,则除数右移一位。得到位移数。

      2.3. 试商,除数-被除数*N = 余数中间值 ,其中N*被除数 <= 除数 && (N+1)*被除数 > 除数。商 = 商 + N * 基数^位移数。

      2.4. 循环执行上述步骤,直到无需再执行高位对齐,那么2.2中得到的余数中间值将作为除法运算的最终余数,否则余数中间值则作为一下轮高位对齐的被除数处理。

      2.5. 符号乘积乘以商得到最终商,符号乘积乘以余数得到最终余数。

 C语言实现:

#include <stdio.h>

// 前置条件
const char lowest_bit_weight = 1; // 二进制最低位的位权重

int main(){
  // 输入
  char dividend = -5, divisor = -2;
 
  // 输出
  char quotients = 0,  // 商
    rem = 0;        // 余数

  // 中间值
  char highest_bit_weight,
       divisor_aligned,
       tmp_dividend = dividend,
       tmp_divisor = divisor;
  char high_alignment;
  char sign,
       sign1 = 1,
       sign2 = 1;
 
  // 负数转换为正数, 求符号乘积
  if (tmp_dividend < 0){
    tmp_dividend = ~tmp_dividend;
    tmp_dividend += 1;
    sign1 = ~sign1;
    sign1 += 1;
  }
  if (tmp_divisor < 0){
    tmp_divisor = ~tmp_divisor;
    tmp_divisor += 1;
    sign2 = ~sign2;
    sign2 += 1;
  }
  sign = sign1 * sign2; 


  // 开始运算
  while (1){
      // 高位对齐 (从高位开始运算)
      // 结果:1. 要么被除数的最高位小于除数的最高位;
      //       2. 要么被除数的最高位对齐除数的最高位, 且被除数大于除数;
          high_alignment = 0;
          highest_bit_weight = lowest_bit_weight;
      divisor_aligned = tmp_divisor;
      while (tmp_dividend >= divisor_aligned){
        divisor_aligned = divisor_aligned << 1;
        highest_bit_weight = highest_bit_weight << 1;

            high_alignment += 1;
      }
      if (high_alignment > 0){
        divisor_aligned = divisor_aligned >> 1;
        highest_bit_weight = highest_bit_weight >> 1;
            high_alignment -= 1;
          }

          // 当无需执行高位对齐时,则将下一轮的被除数作为余数,并且结束运算
          if (0 == high_alignment) {
        rem = tmp_dividend;
        break;
      }

      // 上一轮运算的商加上最高位权重得到当前运算的商值
      quotients = quotients | highest_bit_weight;
      // 被除数减除数的差值作下一轮的被除数
      tmp_dividend = tmp_dividend - divisor_aligned;
  }
  // 商*符号乘积 = 最终商,余数*符号乘积 = 最终余数
  printf("%d/%d=%d(rem:%d)\n", dividend, divisor, quotients * sign, rem * sign);
  return 0;
}

Convert Unsigned 2 Signed, and Convert Signed 2 Unsigned 

  无符号数与有符号数间转换采取的是位级表示(位模式)不变,解析方式变化引起最终所表示的值变化。

  例如:无符号数15的4bit位模式为1111,强制转换为有符号数时其位模式依然是1111,但实际表示的值则变为-1。

  无符号数转换为有符号数的公式 U2Tw(x) = x - xw-1*2w,其中w表示位数,x表示无符号数的十进制值,x表示无符号数的二进制位模式。

  有符号数转换为无符号数的公式 T2Uw(x) = x + xw-1*2w,其中w表示位数,x表示无符号数的十进制值,x表示无符号数的二进制位模式。

 

  注意:在C语言中若参与运算的两运算数分别是有符号数和无符号数,那么会隐式将有符号数转换为无符号数后再进行运算。

目录
相关文章
|
8月前
|
C#
C#学习相关系列之常用符号介绍
C#学习相关系列之常用符号介绍
|
8月前
|
C语言
c语言编程练习题:7-16 计算符号函数的值
请编写程序计算该函数对任一输入整数的值。
128 0
|
8月前
|
C语言
c语言编程练习题:7-52 求简单交错序列前N项和
c语言编程练习题:7-52 求简单交错序列前N项和
72 0
|
8月前
|
算法 搜索推荐 程序员
C语言第十七练——输出二进制中1的个数
C语言第十七练——输出二进制中1的个数
61 0
|
8月前
|
编译器 C++
【C++14保姆级教程】数位分割符、函数返回值推导
【C++14保姆级教程】数位分割符、函数返回值推导
124 0
|
8月前
|
C语言
c语言编程练习题:7-32 求交错序列前N项和
c语言编程练习题:7-32 求交错序列前N项和
141 0
|
8月前
|
C语言
c语言编程练习题:7-56 求给定精度的简单交错序列部分和
c语言编程练习题:7-56 求给定精度的简单交错序列部分和
104 0
|
编译器 C++
c++中基本类型详细解释外加基本运算规则
类型 含义 wchat_t 宽字符 bool 布尔类型 char 字符 chat16_t unicode字符 chat_32 unicode字符 short 短整型 int 整形 long 长整型 longlong 长整型 float 单精度浮点型 double 双精度浮点型 longdouble 扩展精度浮点型
128 1
|
测试技术
(C语言)经典例题之特殊整数求和与方形图案
(C语言)经典例题之特殊整数求和与方形图案
(C语言)经典例题之特殊整数求和与方形图案