问题描述  

  给定两个整数,被除数 dividend 和除数 divisor。将两数相除,要求不使用乘法、除法和 mod 运算符。

  返回被除数 dividend 除以除数 divisor 得到的商。

  示例 1:

  输入: dividend = 10, divisor = 3
  输出: 3

  示例 2:

  输入: dividend = 7, divisor = -3
  输出: -2

  说明:

  • 被除数和除数均为 32 位有符号整数。
  • 除数不为 0。
  • 假设我们的环境只能存储 32 位有符号整数,其数值范围是 [−231,  231 − 1]。本题中,如果除法结果溢出,则返回 231 − 1。

这个问题涉及到了计算机如何利用逻辑运算和加减法来求得除法,这个问题之前一度困扰了我很久。

  

	/**
* 逼近
* 先定符号
* 结果是正是负还是0?
* @param dividend
* @param divisor
* @return
*/
public static int divide(int dividend, int divisor) {
if (dividend == 0) {
return 0;
}
if (dividend == Integer.MIN_VALUE && divisor == -1) {
return Integer.MAX_VALUE;
}
if (divisor == -1) {
return -dividend;
}
if (divisor == 1) {
return dividend;
}
int res = 0, Sum = 0;
    boolean plus = false;
     //同号
    if ((dividend > 0 && divisor > 0) || (dividend < 0 && divisor < 0)) {
      plus = true;
     }
   //将两个数都变成负数
   dividend = dividend > 0 ? ~dividend + 1 : dividend;
   divisor = divisor > 0 ? ~divisor + 1 : divisor;
  for (int i = 30; i > -1; i--) {
       //被加数未溢出 加之后的结果未溢出 加之后的结果小于被除数
  int addNum = divisor<<i;
  if (addNum>>i == divisor && Sum + addNum < 0 && Sum + addNum >= dividend) {
  Sum += addNum;
  res += plus ? 1<<i : -1<<i;
  }
  }return res;
}

  

   题解(写代码时候的奇怪想法。。。):

    首先进行边界处理之类的。

    一开始我采用了二分法猜数字,首先做一个记号记录结果,然后把被除数(dividend)和除数(divisor)都转换为正数 (信息加工),这样结果就一定是在[0,dividend]。

    初始化left = 0, right = dividend,mid = dividend << 1;

    然后通过for循环累加mid次验证是否符合结果( mid * divisor <= dividend并无法取到比mid更大的mid'去满足前面条件);

    由于平时并不是经常使用二分碰到了以下问题:

      二分的边界问题:

        如何写出不杂乱的代码?

        因为常常使用 mid = (left + right)<<1;

        故而遗忘了mid还可以向右偏 mid = ((left + right)<<1) + 1;。

      因为要保证结果一定在边界内,故而

        left = mid + 1; right = mid - 1;常常不能同时出现(视情况而定吧)。

      所以有时候

        采用 right = mid - 1; left = mid;这个组合时:

        mid = (left + right)<<1; (left + 1 = right) 时候回卡死!

        这个时候要mid  = ((left + right)<<1) + 1;(向右偏)

  当然这样的思路写出来的代码的结果就是我挂了。。

    for循环累加代替乘法实在太慢了!!!

  然鹅,这时候我想到了一个办法。

    divisor * mid 可以写成 divisor (m0 * 2^31 + m1 * 2^30 + m2 * 2^31 ....+m30 * 2^0)

    然后二的m次方这个东西我是可以通过左移来得到的!

    于是我兴奋地用这个方法验证mid对不对。

  发现很多边界问题无法解决

  例如:

    之前说的右偏碰到Integer.MIN_VALUE

    Integer.MIN_VALUE无法转换为正数

    mid取得太大,数据溢出,本来divisor * mid已经超过了Integer.MAX_VALUE。却还是几千。。。

  于是我处于崩溃的边缘。。。

    这样搞下去我要屎了!

  然鹅,解手的时候。我想:

    我可以把所有数都转换为负数先啊

    我可以不用猜测mid是多少啊

    我直接从一步一步逼近被除数就行啦??? 好像真的是。。

    例如 :

      结果如若为101010111...(32位)

    那么我从头开始的非符号位开始看能不能加进去就好啦! 如若能加进去就逼近了被除数,数据本身溢出,加进去溢出,加进去大于被除数就代表不能加!

    其他的都加,反正我要的也是最逼近的数。。。

leetcode 29 两数相除的更多相关文章

  1. Java实现 LeetCode 29 两数相除

    29. 两数相除 给定两个整数,被除数 dividend 和除数 divisor.将两数相除,要求不使用乘法.除法和 mod 运算符. 返回被除数 dividend 除以除数 divisor 得到的商 ...

  2. Leetcode 29.两数相除 By Python

    给定两个整数,被除数 dividend 和除数 divisor.将两数相除,要求不使用乘法.除法和 mod 运算符. 返回被除数 dividend 除以除数 divisor 得到的商. 示例 1: 输 ...

  3. LeetCode 29 - 两数相除 - [位运算]

    题目链接:https://leetcode-cn.com/problems/divide-two-integers/description/ 给定两个整数,被除数 dividend 和除数 divis ...

  4. [LeetCode]29 两数相除和一个小坑点

    给定两个整数,被除数 dividend 和除数 divisor.将两数相除,要求不使用乘法.除法和 mod 运算符. 返回被除数 dividend 除以除数 divisor 得到的商. 示例 1: 输 ...

  5. LeetCode 29——两数相除

    1. 题目 2. 解答 2.1. 方法一 题目要求不能使用乘法.除法和除余运算,但我们可以将除法转移到对数域. \[ \frac{a}{b} = e^{\frac{lna}{lnb}} = e^{ln ...

  6. leetcode 29两数相除

    我理解本题是考察基于加减实现除法,代码如下: class Solution { public: //只用加减号实现除法, //不用加减号实现除法: int divide(int dividend, i ...

  7. 【剑指 Offer II 001. 整数除法】同leedcode 29.两数相除

    剑指 Offer II 001. 整数除法 解题思路 在计算的时候将负数转化为正数,对于32位整数而言,最小的正数是-2^31, 将其转化为正数是2^31,导致溢出.因此将正数转化为负数不会导致溢出. ...

  8. [LeetCode] 29. Divide Two Integers 两数相除

    Given two integers dividend and divisor, divide two integers without using multiplication, division ...

  9. LeetCode(29): 两数相除

    Medium! 题目描述: 给定两个整数,被除数 dividend 和除数 divisor.将两数相除,要求不使用乘法.除法和 mod 运算符. 返回被除数 dividend 除以除数 divisor ...

随机推荐

  1. 从零开始一起学习SALM-ICP原理及应用

    点"计算机视觉life"关注,星标更快接收干货! ## 小白:师兄,最近忙什么呢,都见不到你人影,我们的课也好久没更新了呢 师兄:抱歉,抱歉,最近忙于俗事.我后面一起补上,学习劲头 ...

  2. RQNOJ193 造路行动

    题目转移 详见最小生成树讲解 Kruskal #include<cstdio> #include<algorithm> using namespace std; ; int n ...

  3. SQL Server Update 链接修改和when的应用

    一.自链接方式 update b1 set b1.money = b1.money + b2.money from (select * from wallet where type='余额') b1 ...

  4. linux服务器无telnet等测试工具,测试http+json服务连通性

    1. 问题描述: 1.公司内部服务器需要通过http接口方式访问另一公司内部接口服务器. 2.申请信息安全开通访问权限,但是只能开通到服务器+端口号,例如:192.168.1:8080,无ping权限 ...

  5. Hash的应用2

    代码: #include <stdio.h> #define OFFSET 500000//偏移量 ];//记录每个数是否出现,出现为1,不出现为0 int main(){ int n,m ...

  6. 最新 centos7下zabbix 监控redis状态

    准备 先准备两台服务器,已经部署好了zabbix-sever和zabbix-agent zabbix-server 192.168.1.70 centos7.2 zabbix-agent 192.16 ...

  7. Java编程思想:File类list()方法

    import java.util.regex.Pattern; import java.io.File; import java.io.FilenameFilter; public class Tes ...

  8. py+selenium 直接给日期赋值,控制台调试报错【已解决】

    目标:给带日期控件的输入框赋值. 百度去搜索让你各种去只读readonly属性,再send_keys 方法: 其实既然可以去除readonly属性,那就可以直接给属性赋值,将两行代码缩为一行. dri ...

  9. python爬虫笔记之爬取足球比赛赛程

    目标:爬取某网站比赛赛程,动态网页,则需找到对应ajax请求(具体可参考:https://blog.csdn.net/you_are_my_dream/article/details/53399949 ...

  10. 我这边测了一下,发现#后面参数变化浏览器不会刷新,但是#一旦去掉就会刷新了,你那边的url拼的时候能不能在没参数的时候#也拼在里面,这样应该就OK了

    我这边测了一下,发现#后面参数变化浏览器不会刷新,但是#一旦去掉就会刷新了,你那边的url拼的时候能不能在没参数的时候#也拼在里面,这样应该就OK了