Divide two integers without using multiplication, division and mod operator.
描述
不能使用乘法、除法和取模(mod)等运算,除开两个数得到结果,如果内存溢出则返回Integer类型的最大值。解释一下就是:输入两个数,第一个数是被除数dividend,第二个是除数divisor,要求是在不得使用乘法、除法和取模(mod)等运算的前提下,求出两个数的相除结果。
思路
有一个最简单直观的方法,设置一个i=1,比较dividend和divisor大小,如果满足dividend>divisor,就令divisor=divisor+divisor,i++,继续判断dividend>divisor,继续执行循环,直到不满足条件时候输出i的大小。
这种方法最简单直观,但是想想也直到,这个肯定在处理时间上超时。
那么还有什么方法类似于乘除法呢,想到了一种,移位。例如:现在定义一个数int i=2,那么如果执行i=i<<1,那么此时的i就等于4,相当于乘以二了。为了快速得到结果,我们可以加入移位运算,怎么用?用在哪?
现在我们提出一种方法,举个简单的例子:假设输入了两个数,被除数为13,除数为2,下面我们一步一步的解释这种方法:
1、让dividend=13,divisor=2。设一个全局变量的result=0,存储最终结果,设置一个临时结果为re=1,现在这个1表示divisor的值的大小,等于一个原始除数。
2、判断dividend是否大于divisor左移一位,即dividend > divisor<< 是否成立?成立则说明除数大小左移一位(即乘以2)也不会超过被除数,那么执行第3步,否则,执行第4步。
3、re=re<<,divisor=divisor<<,返回执行第2步。
4、判断当前被除数减去当前的除数是否大于等于原始的除数,即判断dividend-divisor>=2是否成立,成立,则说明还有再除以2的空间,只不过现在的除数已经变了,被除数的一部分也已经被除过了,那么就执行第5步,否则,执行第6步;如果不成立,执行第6步。
5、先把临时结果re的值加到最终结果result中去,然后让dividend=dividend-divisor,然后把divisor复位,即让divisor=2,重新执行第2步。
6、返回结果值result。
下面是每一步时,被除数dividend、除数divisor、中间结果re和最终结果result的状态:
步骤 dividend divisor re result
1 13 2 1 0
2 13 2<<=4 1<<=2 0
3 13 4<<=8 2<<=4 0
4 13-8=5 复位为2 复位为1 re+result=4+0=4
5 5 2<<=4 1<<=2 4
6 5-4=1 复位为2 复位为1 re+result=2+4=6
然后就比较不了了,那么结果为6,bingo!
下面贴代码:
public static int divide(int dividend, int divisor) {
//特殊情况:当除数为0时,一定是内存溢出
if(divisor==0) return Integer.MAX_VALUE;
//判断符号位
int sign = 1;
if((dividend>0 && divisor<0) || (dividend<0 && divisor>0)) sign = -1;
//定义为长整型的目的是为了在取绝对值时不会内存溢出
//sor1作为局部的除数出现
long dend = Math.abs((long)dividend), sor = Math.abs((long)divisor), sor1=sor;
//特殊情况:当被除数为0或者被除数小于除数时,返回值一定是0
if(dividend==0 || dend<sor) return 0;
//把最终结果定义成全局变量
long result = 0;
while(true){
//定义一个中间结果,用于对全局结果的累加
long re = 1;
//如果当前的被除数还大于当前除数左移一位之后的数(即乘以2),那么进入循环
while(dend > sor1<<1){
//符合条件情况下,除数左移一位,中间结果也左移一位(中间结果的大小 等于 当前的sor1/sor的结果数)
re=re<<1;
sor1=sor1<<1;
}
result = result+re;
//此时说明中间除数已经到了一个临界点,即当前除数小于当前被除数,但左移一位就会大于被除数。
//判断dend与当前除数之间的差值是否大于等于原始除数,如果等于,说明还有除的空间,那么更新除数和被除数之后重返循环,否则结束
if((dend-sor1) >= sor){
dend=dend-sor1;
sor1=sor;
}else break;
}
//判断内存溢出
if(result*sign>Integer.MAX_VALUE || result*sign<Integer.MIN_VALUE) return Integer.MAX_VALUE;
return (int)result*sign;
}
Divide two integers without using multiplication, division and mod operator.的更多相关文章
- algorithm@ Divide two integers without using multiplication, division and mod operator. (Bit Operation)
#include<bits/stdc++.h> using namespace std; int divide(int dividend, int divisor) { long long ...
- leetcode 29-> Divide Two Integers without using multiplication, division and mod operator
class Solution(object): def divide(self, dividend, divisor): """ :type dividend: int ...
- [LeetCode] Divide Two Integers 两数相除
Divide two integers without using multiplication, division and mod operator. If it is overflow, retu ...
- Leetcode Divide Two Integers
Divide two integers without using multiplication, division and mod operator. 不用乘.除.求余操作,返回两整数相除的结果,结 ...
- leetcode-【中等题】Divide Two Integers
题目 Divide two integers without using multiplication, division and mod operator. If it is overflow, r ...
- [LintCode] Divide Two Integers 两数相除
Divide two integers without using multiplication, division and mod operator. If it is overflow, retu ...
- 62. Divide Two Integers
Divide Two Integers Divide two integers without using multiplication, division and mod operator. 思路: ...
- Divide Two Integers leetcode
题目:Divide Two Integers Divide two integers without using multiplication, division and mod operator. ...
- Java for LeetCode 029 Divide Two Integers
Divide two integers without using multiplication, division and mod operator. If it is overflow, retu ...
随机推荐
- axios api封装
import axios from 'axios'; import qs from 'qs'; const host = 'url' axios.defaults.baseURL = host; // ...
- 关于instanceof的使用
测试单独一个类: <?php class A { } $a = new A(); if($a instanceof A){ echo "对象\$a实现了A类"; } 当一个子 ...
- 【 es搜索】
地图搜索实现: ①参数: 左下角经纬度和右上角经纬度 图层数(zoom) 关键字等各种数据库中的字段 排序方式 具体的坐标点+距离 ②实现 a.用es作为关系库,首先先mapping所有的字段,然后用 ...
- python的基本知识
1. python的简介 python的创始⼈人为吉多·范罗苏姆(Guido van Rossum).1989年年的圣诞节期间,吉多· 范罗苏姆为了了在阿姆斯特丹丹打发时间,决⼼心开发⼀个新的脚 ...
- Java实例 Part2:Java语言基础
Part2:Java语言基础 ** Example01:从控制台接收输入字符 ** 运行结果: 实现代码: import java.util.Scanner; public class Example ...
- Linux3.5—IIC学习分析
I2C控制器的设备对象内核已经实现并关联到platform总线. I2C控制器的驱动对象内核已经实现. 看mach-tiny4412.h /plat-samsung/目录下 /drivers/i2c/ ...
- 在树莓派上用 python 做一个炫酷的天气预报
教大家如何在树莓派上自己动手做一个天气预报.此次教程需要大家有一定的python 基础,没有也没关系,文末我会放出我已写好的代码供大家下载. 首先在开始之前 需要申请高德地图API,去高德地图官网注册 ...
- python教程(二)·数据类型
数据类型,顾名思义就是数据的类型,数据到底有哪些类型呢?在python中,最基本的数据类型有好几种,让我来逐个介绍. 整数类型(int) python中只有一种整数类型,就是int类型,在程序中的写法 ...
- 嵌入式框架Zorb Framework搭建一:嵌入式环境搭建、调试输出和建立时间系统
我是卓波,我是一名嵌入式工程师,我万万没想到我会在这里跟大家吹牛皮. 嵌入式框架Zorb Framework搭建过程 嵌入式框架Zorb Framework搭建一:嵌入式环境搭建.调试输出和建立时间系 ...
- 分享Centos6.5升级glibc过程
默认的Centos6.5 glibc版本最高为2.12, 而在进行Nodejs开发时项目所依赖的包往往需要更高版本的glibc库支持, 因此在不升级系统的前提下, 需要主动更新系统glibc库. 一般 ...