听同学百度二面中,不准用四则运算操作符来实现四则运算。一想就想到了计算机组成原理上学过的。位运算的思想可以应用到很多地方,这里简单的总结一下用位运算来实现整数的四则运算。

加法运算:

[cpp] view plaincopy
  1. int AddWithoutArithmetic(int num1,int num2)
  2. {
  3. if(num2==0) return num1;//没有进位的时候完成运算
  4. int sum,carry;
  5. sum=num1^num2;//完成第一步没有进位的加法运算
  6. carry=(num1&num2)<<1;//完成第二步进位并且左移运算
  7. return AddWithoutArithmetic(sum,carry);//进行递归,相加
  8. }

简化一下:

[cpp] view plaincopy
  1. int Add(int a,int b)
  2. {
  3. return b ? Add(a^b,(a&b)<<1) : a;
  4. /*if(b)
  5. return Add(a^b,(a&b)<<1);
  6. else
  7. return a;*/
  8. }

上面的思路就是先不计进位相加,然后再与进位相加,随着递归,进位会变为0,递归结束。

非递归的版本如下:

[cpp] view plaincopy
  1. int Add(int a, int b)
  2. {
  3. int ans;
  4. while(b)
  5. {   //直到没有进位
  6. ans = a^b;        //不带进位加法
  7. b = ((a&b)<<1);   //进位
  8. a = ans;
  9. }
  10. return a;
  11. }

减法运算:

[cpp] view plaincopy
  1. //这个和加法一样了,首先取减数的补码,然后相加。
  2. int negtive(int a)   //取补码
  3. {
  4. return Add(~a, 1);
  5. }
  6. int Sub(int a, int b)
  7. {
  8. return Add(a, negtive(b));
  9. }

正数乘法运算:

[cpp] view plaincopy
  1. //正数乘法运算
  2. int Pos_Multiply(int a,int b)
  3. {
  4. int ans = 0;
  5. while(b)
  6. {
  7. if(b&1)
  8. ans = Add(ans, a);
  9. a = (a<<1);
  10. b = (b>>1);
  11. }
  12. return ans;
  13. }

整数除法(正整数)

[cpp] view plaincopy
  1. //除法就是由乘法的过程逆推,依次减掉(如果x够减的)y^(2^31),y^(2^30),...y^8,y^4,y^2,y^1。减掉相应数量的y就在结果加上相应的数量。
  2. int Pos_Div(int x,int y)
  3. {
  4. int ans=0;
  5. for(int i=31;i>=0;i--)
  6. {
  7. //比较x是否大于y的(1<<i)次方,避免将x与(y<<i)比较,因为不确定y的(1<<i)次方是否溢出
  8. if((x>>i)>=y)
  9. {
  10. ans+=(1<<i);
  11. x-=(y<<i);
  12. }
  13. }
  14. return ans;
  15. }

完整的实现:

[cpp] view plaincopy
  1. // 加减乘除位运算
  2. // 程序中实现了比较大小、加减乘除运算。所有运算都用位操作实现
  3. // 在实现除法运算时,用了从高位到低位的减法
  4. // 具体如下,算法也比较简单,所以没有作注释
  5. #include<iostream>
  6. #include<cstdio>
  7. using namespace std;
  8. int Add(int a, int b)
  9. {
  10. int ans;
  11. while(b)
  12. {  //直到没有进位
  13. ans = a^b;        //不带进位加法
  14. b = ((a&b)<<1);   //进位
  15. a = ans;
  16. }
  17. return a;
  18. }
  19. //这个和加法一样了,首先取减数的补码,然后相加。
  20. int negtive(int a)   //取补码
  21. {
  22. return Add(~a, 1);
  23. }
  24. int Sub(int a, int b)
  25. {
  26. return Add(a, negtive(b));
  27. }
  28. // 判断正负
  29. int ispos( int a )
  30. { //正
  31. return (a&0xFFFF) && !(a&0x8000);
  32. }
  33. int isneg( int a )
  34. { //负
  35. return a&0x8000;
  36. }
  37. int iszero( int a )
  38. { //0
  39. return !(a&0xFFFF);
  40. }
  41. //正数乘法运算
  42. int Pos_Multiply(int a,int b)
  43. {
  44. int ans = 0;
  45. while(b)
  46. {
  47. if(b&1)
  48. ans = Add(ans, a);
  49. a = (a<<1);
  50. b = (b>>1);
  51. }
  52. return ans;
  53. }
  54. //乘法运算
  55. int Multiply(int a,int b)
  56. {
  57. if( iszero(a) || iszero(b) )
  58. return 0;
  59. if( ispos(a) && ispos(b) )
  60. return Pos_Multiply(a, b);
  61. if( isneg(a) )
  62. {
  63. if( isneg(b) )
  64. {
  65. return Pos_Multiply( negtive(a), negtive(b) );
  66. }
  67. return negtive( Pos_Multiply( negtive(a), b ) );
  68. }
  69. return negtive( Pos_Multiply(a, negtive(b)) );
  70. }
  71. //除法就是由乘法的过程逆推,依次减掉(如果x够减的)y^(2^31),y^(2^30),...y^8,y^4,y^2,y^1。减掉相应数量的y就在结果加上相应的数量。
  72. int Pos_Div(int x,int y)
  73. {
  74. int ans=0;
  75. for(int i=31;i>=0;i--)
  76. {
  77. //比较x是否大于y的(1<<i)次方,避免将x与(y<<i)比较,因为不确定y的(1<<i)次方是否溢出
  78. if((x>>i)>=y)
  79. {
  80. ans+=(1<<i);
  81. x-=(y<<i);
  82. }
  83. }
  84. return ans;
  85. }
  86. //除法运算
  87. int MyDiv( int a, int b )
  88. {
  89. if( iszero(b) )
  90. {
  91. cout << "Error" << endl;
  92. exit(1);
  93. }
  94. if( iszero(a) )
  95. return 0;
  96. if( ispos(a) )
  97. {
  98. if( ispos(b) )
  99. return Pos_Div(a, b);
  100. return negtive( Pos_Div( a, negtive(b)) );
  101. }
  102. if( ispos(b) )
  103. return negtive( Pos_Div( negtive(a), b ) );
  104. return Pos_Div( negtive(a), negtive(b) );
  105. }
  106. // 比较两个正数的大小(非负也可)
  107. int isbig_pos( int a, int b )
  108. {  //a>b>0
  109. int c = 1;
  110. b = (a^b);
  111. if( iszero(b) )
  112. return 0;
  113. while( b >>= 1 )
  114. {
  115. c <<= 1;
  116. }
  117. return (c&a);
  118. }
  119. // 比较两个数的大小
  120. int isbig( int a, int b )
  121. { //a>b
  122. if( isneg(a) )
  123. {
  124. if( isneg(b) )
  125. {
  126. return isbig_pos( negtive(b), negtive(a) );
  127. }
  128. return 0;
  129. }
  130. if( isneg(b) )
  131. return 1;
  132. return isbig_pos(a, b);
  133. }

扩展:在不使用*、/、+、-、%操作符的情况下,如何求一个数的1/3?(用C语言实现)
使用位操作符并实现“+”操作

[cpp] view plaincopy
  1. // 替换加法运算符
  2. int add(int x , int y)
  3. {
  4. int res;
  5. while(y)       // 直到没有进位
  6. {
  7. res = x^y;       // 不带进位的加法
  8. y = ((x&y)<<1);  // 进位
  9. x = res;
  10. }
  11. return x;
  12. }
  13. int divideby3(int num)
  14. {
  15. int sum = 0;
  16. while(num > 3)
  17. {
  18. sum = add(num>>2 , sum);
  19. num = add(num>>2 , num&3);
  20. }
  21. if(num == 3)
  22. sum = add(sum , 1);
  23. return sum;
  24. }

原理: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

用位运算实现加减乘除相关推荐

  1. C++学习笔记-----用位运算实现加减乘除

    C++学习笔记-----用位运算实现加减乘除 原文:http://blog.csdn.net/sinat_35261315/article/details/72904945 数据在计算机内存中是以二进 ...

  2. 【C++】位运算实现加减乘除

    1 #include<iostream> 2 #include<assert.h> 3 using namespace std; 4 5 // 位运算实现加减乘除 6 7 in ...

  3. java加减_Java位运算实现加减乘除

    一.加法 a+b 举例实现:13+9=22 13+9不考虑进位结果为12 只考虑进位结果为10 和刚好是22. 13二进制为1101,9二进制为1001. 不考虑进位结果为0100.算式为a^b 只考 ...

  4. 只使用位运算实现加减乘除

    在线OJ: LeetCode 29. 两数相除 原题目的要求是不能使用乘法, 除法和取余运算符实现除法. 在本篇博客中把题目要求提高一点, 这里只使用位运算来实现, 顺便的也就把只使用位运算实现加减乘 ...

  5. 数学问题——位运算实现加减乘除

    ##位运算实现加减乘除 ###位运算实现加法 public static int add(int a, int b) {int sum = a;while (b != 0) {//不断的相加,直到没有 ...

  6. Java 实现位运算计算加减乘除

    机器数和机器数的真值 一个数在计算机中的二进制表示形式,叫做这个数的机器数.机器数是带符号的,在计算机用机器数的最高位存放符号,正数为0,负数为1.举个例子,比如在机器字长为8位的情况下(机器字长是指 ...

  7. 位运算实现加减乘除运算

    我们知道,计算机最基本的操作单元是字节(byte),一个字节由8个位(bit)组成,一个位只能存储一个0或1,其实也就是高低电平.无论多么复杂的逻辑.庞大的数据.酷炫的界面,最终体现在计算机最底层都只 ...

  8. java用位运算实现加减乘除

    个人主页:熬夜磕代码丶 作品专栏: java se 我变秃了,也变强了 给大家介绍一款程序员必备刷题平台--牛客网 点击注册一起刷题收获大厂offer吧 文章目录 前言 一.常见位运算 1. & ...

  9. java实现三位数加减乘除_用Java位运算实现加减乘除四则运算

    感谢博客:http://blog.csdn.net/itismelzp/article/details/49621741  提供的思路. 要用位运算来实现四则运算,不仅仅要知道&,|,~,^, ...

最新文章

  1. SQLite基本操作
  2. 免费有理之文件备份软件
  3. 机器学习论文泛读总结
  4. Saturn分布式调度之系统架构简介
  5. Java:转换列表 String 到一个字符串
  6. 开始使用Filebeat
  7. 三网融合:不给安全营造“盗梦空间”
  8. java程序设计精编教程第3版电子版课后答案_Java程序设计精编教程(第3版)-微课版...
  9. 中e管家理财类型都有哪些
  10. Python初学者:输入若干整数,输出最大值及其索引
  11. 试用74LS194加74151设计一个从Q0端输出100111序列信号的序列信号发生器. 要求电路能自启动, 且越简单越好
  12. root 红米note5_怎么开启红米Note 5的ROOT权限
  13. 网络流媒体(四)———TS流
  14. 冷库用电量计算机方式,冷库耗电量计算方法(附计算公式)
  15. 操作系统笔记 第二章
  16. 【pandas drop()和dropna()函数使用详解】
  17. 【js】vue项目中实现点击复制过滤条件,获取并处理粘贴板内容
  18. 电脑一个磁盘分为两个磁盘
  19. 请教知网检索返回网址的问题
  20. 天下数据教你提升网站访问速度的妙招

热门文章

  1. jenkins+ant+git+android搭建笔记
  2. SQL Server2008附加数据库之后显示为只读时解决方法
  3. 没事试试50mm1.4
  4. REPAIR修复mysql报错is marked as crashed and should be
  5. 使用hexo yeele主题搭建个人博客
  6. Single-Shot Object Detection with Enriched Semantics
  7. 雅虎宣布关闭旗下7个数字杂志
  8. HTML5----简易贪吃蛇小游戏
  9. QT STUDY 模型-视图-控制器
  10. Errors occurred during the build