用位运算实现加减乘除
听同学百度二面中,不准用四则运算操作符来实现四则运算。一想就想到了计算机组成原理上学过的。位运算的思想可以应用到很多地方,这里简单的总结一下用位运算来实现整数的四则运算。
加法运算:
- int AddWithoutArithmetic(int num1,int num2)
- {
- if(num2==0) return num1;//没有进位的时候完成运算
- int sum,carry;
- sum=num1^num2;//完成第一步没有进位的加法运算
- carry=(num1&num2)<<1;//完成第二步进位并且左移运算
- return AddWithoutArithmetic(sum,carry);//进行递归,相加
- }
简化一下:
- int Add(int a,int b)
- {
- return b ? Add(a^b,(a&b)<<1) : a;
- /*if(b)
- return Add(a^b,(a&b)<<1);
- else
- return a;*/
- }
上面的思路就是先不计进位相加,然后再与进位相加,随着递归,进位会变为0,递归结束。
非递归的版本如下:
- int Add(int a, int b)
- {
- int ans;
- while(b)
- { //直到没有进位
- ans = a^b; //不带进位加法
- b = ((a&b)<<1); //进位
- a = ans;
- }
- return a;
- }
减法运算:
- //这个和加法一样了,首先取减数的补码,然后相加。
- int negtive(int a) //取补码
- {
- return Add(~a, 1);
- }
- int Sub(int a, int b)
- {
- return Add(a, negtive(b));
- }
正数乘法运算:
- //正数乘法运算
- int Pos_Multiply(int a,int b)
- {
- int ans = 0;
- while(b)
- {
- if(b&1)
- ans = Add(ans, a);
- a = (a<<1);
- b = (b>>1);
- }
- return ans;
- }
整数除法(正整数)
- //除法就是由乘法的过程逆推,依次减掉(如果x够减的)y^(2^31),y^(2^30),...y^8,y^4,y^2,y^1。减掉相应数量的y就在结果加上相应的数量。
- int Pos_Div(int x,int y)
- {
- int ans=0;
- for(int i=31;i>=0;i--)
- {
- //比较x是否大于y的(1<<i)次方,避免将x与(y<<i)比较,因为不确定y的(1<<i)次方是否溢出
- if((x>>i)>=y)
- {
- ans+=(1<<i);
- x-=(y<<i);
- }
- }
- return ans;
- }
完整的实现:
- // 加减乘除位运算
- // 程序中实现了比较大小、加减乘除运算。所有运算都用位操作实现
- // 在实现除法运算时,用了从高位到低位的减法
- // 具体如下,算法也比较简单,所以没有作注释
- #include<iostream>
- #include<cstdio>
- using namespace std;
- int Add(int a, int b)
- {
- int ans;
- while(b)
- { //直到没有进位
- ans = a^b; //不带进位加法
- b = ((a&b)<<1); //进位
- a = ans;
- }
- return a;
- }
- //这个和加法一样了,首先取减数的补码,然后相加。
- int negtive(int a) //取补码
- {
- return Add(~a, 1);
- }
- int Sub(int a, int b)
- {
- return Add(a, negtive(b));
- }
- // 判断正负
- int ispos( int a )
- { //正
- return (a&0xFFFF) && !(a&0x8000);
- }
- int isneg( int a )
- { //负
- return a&0x8000;
- }
- int iszero( int a )
- { //0
- return !(a&0xFFFF);
- }
- //正数乘法运算
- int Pos_Multiply(int a,int b)
- {
- int ans = 0;
- while(b)
- {
- if(b&1)
- ans = Add(ans, a);
- a = (a<<1);
- b = (b>>1);
- }
- return ans;
- }
- //乘法运算
- int Multiply(int a,int b)
- {
- if( iszero(a) || iszero(b) )
- return 0;
- if( ispos(a) && ispos(b) )
- return Pos_Multiply(a, b);
- if( isneg(a) )
- {
- if( isneg(b) )
- {
- return Pos_Multiply( negtive(a), negtive(b) );
- }
- return negtive( Pos_Multiply( negtive(a), b ) );
- }
- return negtive( Pos_Multiply(a, negtive(b)) );
- }
- //除法就是由乘法的过程逆推,依次减掉(如果x够减的)y^(2^31),y^(2^30),...y^8,y^4,y^2,y^1。减掉相应数量的y就在结果加上相应的数量。
- int Pos_Div(int x,int y)
- {
- int ans=0;
- for(int i=31;i>=0;i--)
- {
- //比较x是否大于y的(1<<i)次方,避免将x与(y<<i)比较,因为不确定y的(1<<i)次方是否溢出
- if((x>>i)>=y)
- {
- ans+=(1<<i);
- x-=(y<<i);
- }
- }
- return ans;
- }
- //除法运算
- int MyDiv( int a, int b )
- {
- if( iszero(b) )
- {
- cout << "Error" << endl;
- exit(1);
- }
- if( iszero(a) )
- return 0;
- if( ispos(a) )
- {
- if( ispos(b) )
- return Pos_Div(a, b);
- return negtive( Pos_Div( a, negtive(b)) );
- }
- if( ispos(b) )
- return negtive( Pos_Div( negtive(a), b ) );
- return Pos_Div( negtive(a), negtive(b) );
- }
- // 比较两个正数的大小(非负也可)
- int isbig_pos( int a, int b )
- { //a>b>0
- int c = 1;
- b = (a^b);
- if( iszero(b) )
- return 0;
- while( b >>= 1 )
- {
- c <<= 1;
- }
- return (c&a);
- }
- // 比较两个数的大小
- int isbig( int a, int b )
- { //a>b
- if( isneg(a) )
- {
- if( isneg(b) )
- {
- return isbig_pos( negtive(b), negtive(a) );
- }
- return 0;
- }
- if( isneg(b) )
- return 1;
- return isbig_pos(a, b);
- }
扩展:在不使用*、/、+、-、%操作符的情况下,如何求一个数的1/3?(用C语言实现)
使用位操作符并实现“+”操作
- // 替换加法运算符
- int add(int x , int y)
- {
- int res;
- while(y) // 直到没有进位
- {
- res = x^y; // 不带进位的加法
- y = ((x&y)<<1); // 进位
- x = res;
- }
- return x;
- }
- int divideby3(int num)
- {
- int sum = 0;
- while(num > 3)
- {
- sum = add(num>>2 , sum);
- num = add(num>>2 , num&3);
- }
- if(num == 3)
- sum = add(sum , 1);
- return sum;
- }
原理:n = 4 * a + b; n / 3 = a + (a + b) / 3; 然后 sum += a, n = a + b 并迭代; 当 a == 0 (n < 4)时,sum += floor(n / 3); i.e. 1, if n == 3, else 0
用位运算实现加减乘除相关推荐
- C++学习笔记-----用位运算实现加减乘除
C++学习笔记-----用位运算实现加减乘除 原文:http://blog.csdn.net/sinat_35261315/article/details/72904945 数据在计算机内存中是以二进 ...
- 【C++】位运算实现加减乘除
1 #include<iostream> 2 #include<assert.h> 3 using namespace std; 4 5 // 位运算实现加减乘除 6 7 in ...
- java加减_Java位运算实现加减乘除
一.加法 a+b 举例实现:13+9=22 13+9不考虑进位结果为12 只考虑进位结果为10 和刚好是22. 13二进制为1101,9二进制为1001. 不考虑进位结果为0100.算式为a^b 只考 ...
- 只使用位运算实现加减乘除
在线OJ: LeetCode 29. 两数相除 原题目的要求是不能使用乘法, 除法和取余运算符实现除法. 在本篇博客中把题目要求提高一点, 这里只使用位运算来实现, 顺便的也就把只使用位运算实现加减乘 ...
- 数学问题——位运算实现加减乘除
##位运算实现加减乘除 ###位运算实现加法 public static int add(int a, int b) {int sum = a;while (b != 0) {//不断的相加,直到没有 ...
- Java 实现位运算计算加减乘除
机器数和机器数的真值 一个数在计算机中的二进制表示形式,叫做这个数的机器数.机器数是带符号的,在计算机用机器数的最高位存放符号,正数为0,负数为1.举个例子,比如在机器字长为8位的情况下(机器字长是指 ...
- 位运算实现加减乘除运算
我们知道,计算机最基本的操作单元是字节(byte),一个字节由8个位(bit)组成,一个位只能存储一个0或1,其实也就是高低电平.无论多么复杂的逻辑.庞大的数据.酷炫的界面,最终体现在计算机最底层都只 ...
- java用位运算实现加减乘除
个人主页:熬夜磕代码丶 作品专栏: java se 我变秃了,也变强了 给大家介绍一款程序员必备刷题平台--牛客网 点击注册一起刷题收获大厂offer吧 文章目录 前言 一.常见位运算 1. & ...
- java实现三位数加减乘除_用Java位运算实现加减乘除四则运算
感谢博客:http://blog.csdn.net/itismelzp/article/details/49621741 提供的思路. 要用位运算来实现四则运算,不仅仅要知道&,|,~,^, ...
最新文章
- SQLite基本操作
- 免费有理之文件备份软件
- 机器学习论文泛读总结
- Saturn分布式调度之系统架构简介
- Java:转换列表 String 到一个字符串
- 开始使用Filebeat
- 三网融合:不给安全营造“盗梦空间”
- java程序设计精编教程第3版电子版课后答案_Java程序设计精编教程(第3版)-微课版...
- 中e管家理财类型都有哪些
- Python初学者:输入若干整数,输出最大值及其索引
- 试用74LS194加74151设计一个从Q0端输出100111序列信号的序列信号发生器. 要求电路能自启动, 且越简单越好
- root 红米note5_怎么开启红米Note 5的ROOT权限
- 网络流媒体(四)———TS流
- 冷库用电量计算机方式,冷库耗电量计算方法(附计算公式)
- 操作系统笔记 第二章
- 【pandas drop()和dropna()函数使用详解】
- 【js】vue项目中实现点击复制过滤条件,获取并处理粘贴板内容
- 电脑一个磁盘分为两个磁盘
- 请教知网检索返回网址的问题
- 天下数据教你提升网站访问速度的妙招
热门文章
- jenkins+ant+git+android搭建笔记
- SQL Server2008附加数据库之后显示为只读时解决方法
- 没事试试50mm1.4
- REPAIR修复mysql报错is marked as crashed and should be
- 使用hexo yeele主题搭建个人博客
- Single-Shot Object Detection with Enriched Semantics
- 雅虎宣布关闭旗下7个数字杂志
- HTML5----简易贪吃蛇小游戏
- QT STUDY 模型-视图-控制器
- Errors occurred during the build