位运算实现加减乘除四则运算(Java)

本文是继《一文了解有趣的位运算》的第二篇文章.

我们知道,计算机最基本的操作单元是字节(byte),一个字节由8个位(bit)组成,一个位只能存储一个0或1,其实也就是高低电平。无论多么复杂的逻辑、庞大的数据、酷炫的界面,最终体现在计算机最底层都只是对0101的存储和运算。因此,了解位运算有助于提升我们对计算机底层操作原理的理解。

一、加法#

两个二进制数异或运算的结果是不考虑进位时的结果,

两个二进制数与运算的结果中含有1的位是有进位的位。

0101 + 0001 = 0110为例分析如下:

 

Copy

//计算 0101 + 0001 0101 ^ 0001 = 0100 //异或结果表明,如果不考虑进位,那么结果为0100 0101 & 0001 = 0001 //与运算结果表明,最低位需要向次低位进1 0001 << 1 = 0010 //与运算结果左移一位,将进位加到高位上 //递归计算 0100 + 0010,直到+号右侧数字为0

java代码:

递归

 

Copy

public static int add(int a, int b) { if (b == 0) { return a; } else { return add(a ^ b, (a & b) << 1); } }

循环

 

Copy

public static int add2(int a, int b) { int sum = a; while (b != 0) { sum = a ^ b; b = (a & b) << 1; a = sum; } return sum; }

二、减法#

与加法的思路一致,只不过减去一个数等于加一个数的相反数。

例如:5-1 = 5+(-1)。所以,我们只需要求出被减数的相反数即可。

如何求出一个数的相反数?

计算机中存储的是二进制的补码形式,正数的补码与原码相同,负数的补码为对该数的原码除符号位外各位取反,然后在最后一位加1。

例如:

1在计算机中的二进制表示为:0000 0001

-1在计算机中的二进制表示为:1111 1111

计算过程为:

-1的原码:1000 0001

-1的反码:1111 1110

-1的补码:1111 1111

其中,由1的原码(0000 0001)取反可得-1的反码(1111 1110)

总结,一个数的相反数的求法就是该数每一位取反,末位加一。

java代码:

 

Copy

public static int minus(int a, int b) { return add(a, add(~b, 1)); }

三、乘法#

如果没有思路,可以先在纸上笔算二进制乘法的过程:

 

Copy

0101 a × 0110 b ---------------- 0000 0101 0101 + 0000 ---------------- 00011110

梳理下笔算二进制乘法的过程:

初始化乘积结果为0,依次遍历数字b的末位 0→1→1→0,当末位为0时,乘积结果加上0,也就是乘积不变,A左移一位;当末位为1时,乘积结果加上A,A再左移一位。

如何遍历数字b的末位呢?

根据前面所学,我们可以使用与运算取一个数的末位,并不断右移数字b,直到数字b==0,即可结束位移。

需要注意的是正负数的符号问题,此处是先对a、b两数的绝对值计算其乘积,最后再确定其符号。

java代码为:

 

Copy

public static int multiply(int a, int b) { //将乘数和被乘数都取绝对值 int A = a < 0 ? add(~a, 1) : a; int B = b < 0 ? add(~b, 1) : b; //计算绝对值的乘积 int P = 0; while (B != 0) { if ((B & 1) != 0) { //取乘数的二进制的最后一位,0 or 1 P = add(P, A); } A = A << 1; B = B >> 1; } //计算乘积的符号 if ((a ^ b) < 0) { P = add(~P, 1); } return P; }

四、除法#

最简单的除法实现就是不停的用除数去减被除数,直到被除数小于除数时,此时所减的次数就是我们需要的商,而此时的被除数就是余数。

唯一需要注意的就是商的符号和余数的符号。商的符号确定方式也乘法是一样,即同号为正,异号为负。而余数的符号和被除数的符号是一样的。

和简单的乘法实现一样,这里我们要先对两数的绝对值求商,求余数。最后再确定符号。

 

Copy

public static int[] divide(int a, int b) { //对被除数和除数取绝对值 int A = a < 0 ? add(~a, 1) : a; int B = b < 0 ? add(~b, 1) : b; //对被除数和除数的绝对值求商 int C = A; // 余数C int N = 0; // 商N while (C >= B) { C = minus(C, B); // C-B N = add(N, 1); // N+1 } // 求商的符号 if ((a ^ b) < 0) { N = add(~N, 1); } // 求余数的符合 if (a < 0) { C = add(~C, 1); } return new int[]{N, C}; }

需要指出的是,这种算法在A很大、B很小的情况下效率很低,那该如何优化算法减少while循环的次数呢?

不难想到,除法是由乘法的过程逆推而来的。例如 9÷4=2...1,也就是2*4+1=9。假设用9去减4*2,可以得出结果等于1,因为1小于4,那么就可以得出9÷4的商是2,余数是1。

如何确定4的倍数是逼近最终结果的关键。我们知道,int 整型有32位,除首位表示符号位,每一位的大小是 [2^0, 2^1, 2^2, , , 2^30],最大的int整数是2^31-1。所以,我们可以依次将被除数与2^31, 2^30, ...2^3, 2^2, 2^1, 1相乘,如果除数大于它们的乘积,除数就与之相减,并用相减得到的余数继续作为除数,直到循环结束。

java代码:

 

Copy

public static int[] divide(int a, int b) { // 对被除数和除数取绝对值 int A = a < 0 ? add(~a, 1) : a; int B = b < 0 ? add(~b, 1) : b; int N = 0; // 商 N for (int i = 31; i >= 0; i--) { // 未使用A>=(B<<i)进行判断,因为只有左移B时舍弃的高位不包含1,才相当于该数乘以2的i次方. if ((A >> i) >= B) { // A ÷ 2^i >= B N += (1 << i); // N = N + 2^i A -= (B << i); // A = A - B*2^i } } int C = A; // 余数C // 求商的符号 if ((a ^ b) < 0) { N = add(~N, 1); } // 求余数的符号 if (a < 0) { C = add(~C, 1); } return new int[]{N, C}; }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值