最近,在百度知道上回答问题,然后看见有的人问如何用位移运算符去进行加减乘除运算,于是巩固今天就在这总结一下。

  先讲讲总体思路:
加法运算:将一个整数用二进制表示,其加法运算就是:相异(^)时,本位为1,进位为0;同为1时本位为0,进位为1;同为0时,本位进位均为0.所以,不计进位的和为sum
= a^b,进位就是arr =
a&b,(与sum相加时先左移一位,因为这是进位)。完成加法直到进位为0.

减法运算:a-b  =
a+(-b)  根据补码的特性,各位取反加1即可(注意得到的是相反数,不是该数的补码,因为符号位改变了)

(上面用二进制实现的加减法可以直接应用于负数)

乘法运算:原理上还是通过加法计算。将b个a相加,注意下面实际的代码。

除法运算:除法运算是乘法的逆。看a最多能减去多少个b。

1、加法:
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);//进行递归,相加
 
}
 
其实这个还可以写得更简单,我们用的思路就是先不计进位相加,然后再与进位相加,随着递归,进位会变为0,递归结束。代码如下:
int Add(int a,int b)
 
{  
   
return b ? Add(a^b,(a&b)<<1) : a;
 
   
 
}  

  当然递归有时候不好理解,所以我有写个不用递归的:
int Add(int a, int b)
 
{  
   
int ans;  
   
while(b)  
    {
  //直到没有进位  
   
    ans = a^b;  
     //不带进位加法
 
   
    b = ((a&b)<<1);
  //进位  
   
    a = ans;
 
    }
 
   
return a;  
}  

2、减法:
//这个和加法一样了,首先取减数的补码,然后相加。
 
int negtive(int a)  
//取补码  
{  
   
return Add(~a, 1);  
}  
int Sub(int a, int b)
 
{  
   
return Add(a, negtive(b));  

3、正数乘法运算:
//正数乘法运算  
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;  
}  
 

4、正整数除法:
//除法就是由乘法的过程逆推,依次减掉(如果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  
#include  
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);  
}  

C语言实现用位移运算符进行加减乘…的更多相关文章

  1. 搬运1:关于对C语言中数组名取地址加减等操作的一点探究

    对于数组名取地址强制转换的操作 偶然在晚上学了C语言指针后网页闲逛找题时,被一个数组名取地址搞糊涂了,在自己试验加探索后我稍微悟了一点东西. 代码如下: #include<stdio.h> ...

  2. 大整数加减运算的C语言实现

    目录 大整数加减运算的C语言实现 一. 问题提出 二. 代码实现 三. 效果验证 大整数加减运算的C语言实现 标签: 大整数加减 C 一. 问题提出 培训老师给出一个题目:用C语言实现一个大整数计算器 ...

  3. 2.2JAVA基础复习——JAVA语言的基础组成运算符和语句

    JAVA语言的基础组成有: 1.关键字:被赋予特殊含义的单词. 2.标识符:用来标识的符号. 3.注释:用来注释说明程序的文字. 4.常量和变量:内存存储区域的表示. 5.运算符:程序中用来运算的符号 ...

  4. Swift语言指南(九)--基本运算符

    原文:Swift语言指南(九)--基本运算符 运算符(operator)是用来检查,改变或合并值的一种特殊符号或短语.例如,加号运算符让两个数字相加(如:let i = 1 + 2),还有些更复杂的运 ...

  5. Java补码表和位移运算符

    在java中数据都是以二进制的形式保存的. 但是我们看到的数据怎么是10进制的? 因为java展示之前会自动调用toString()方法 这里以4位2进制为例,4位2进制只能表示16个数,即0-15. ...

  6. c语言的类型、运算符与表达式

    title: 2017-10-17c语言的类型.运算符与表达式 tags: c程序设计语言 grammar_cjkRuby: true --- 1.1 数据类型 char 字符型,一个字节 int 整 ...

  7. C语言入门教程-(6)运算符

    1.运算符概述 运算符是一种编译器执行特定的数学或逻辑操作的符号.C语言提供了以下类型的运算符: 算术运算符 关系运算符 逻辑运算符 位运算符 赋值运算符 条件运算符 其他运算符 2.算术运算符 算术 ...

  8. Java 中位移运算符 >>,>>>,<<

    Java 中的三种位移运算符 java中有三种移位运算符 <<      :     左移运算符,num << 1,相当于num乘以2 >>      :     ...

  9. 正经学C#_位移与其位移运算符[c#入门经典]

    在c#入门经典一书中,最为糟糕的一节就是位移了,完全没有讲明白,也没有说全,似乎只是轻轻点了一下何为位移,带了两次原码和补码,完全不理会是否明白不明白.这一点这本书很差.因为此书说了,在大多数应用开发 ...

随机推荐

  1. jquery使用FormData提交数据

    在jquery中,使用ajax提交表单数据. FormData可以很方便地获取到表单中的所有数据. 注意: ajax中的data参数为FormData类型时,contentType就不要设置成appl ...

  2. as3.0控制声音大小

    //加载声音 var sound:Sound=new Sound(); sound.load(new URLRequest("1.mp3")): //声明声道 var soundC ...

  3. PAT1131(dfs)

    In the big cities, the subway systems always look so complex to the visitors. To give you some sense ...

  4. Could not get lock /var/lib/dpkg/lock更新问题

    发生于apt-get update或apt update时. 常见手段,先试试: sudo rm /var/lib/dpkg/lock sudo rm /var/lib/apt/lists/lock ...

  5. 在IDEA中使用MyBatis Generator逆向工程生成代码

    本文介绍一下用Maven工具如何生成Mybatis的代码及映射的文件. 一.配置Maven pom.xml 文件 在pom.xml增加以下插件: <build> <finalName ...

  6. 【svn】服务器搭建和迁移

    导语 svn客户端大部分开发都会用到,但是为什么我们仍然需要svn服务端呢? 理由可能有: 1,我们想存放一些属于自己的文档,而不像被其他人发现(在自己的网络环境中,安全性更高,更易用,不依赖于公司, ...

  7. 《基于Nginx的中间件架构》学习笔记---1.环境配置

    一.环境调试确认 (四项确认) 1.确认系统网络 ping www.baidu.com 2.确认yum可用 yum list|grep gcc 3.确认关闭iptables规则 iptables -L ...

  8. Django的Rbac介绍3

    今天的博客主要是记录一下如何实现左侧菜单,这里我们想实现的效果就是,如果用户有查看用户的权限,则显示查看用户的左侧菜单,如果用户有查看角色的权限,则显示查看角色的左侧菜单,如果两者都有,则需要显示两个 ...

  9. 高盛oa

    一道题根本不会,抄答案过了.一道自己写,莫名其妙出现了不会的bug.最后交了暴力解,过了5/7.估计要跪. 总结: 缺点:做过的不熟练.没做过的题不会.一个陌生的小bug也de不出来. 措施:多总结还 ...

  10. Redhat(RHEL)配置静态IP

    vim /etc/sysconfig/network NETWORKING=yes NETWORKING_IPV6=no HOSTNAME=itcc.dev GATEWAY=192.168.0.1 v ...