问题描述  

  给定两个整数,被除数 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. linux c库函数大全

    Linux C函数库参考手册  [转自ChinaUnix]第1章字符测试函数isalnum(测试字符是否为英文字母或数字)isalpha(测试字符是否为英文字母)isascii(测试字符是否为ASCI ...

  2. 分布式事务(1)---2PC和3PC理论

    分布式事务(1)---2PC和3PC理论 分布式事物基本理论:基本遵循CPA理论,采用柔性事物特征,软状态或者最终一致性特点保证分布式事物一致性问题. 分布式事物常见解决方案: 2PC两段提交协议 3 ...

  3. Akka-CQRS(16)- gRPC用JWT进行权限管理

    前面谈过gRPC的SSL/TLS安全机制,发现设置过程比较复杂:比如证书签名:需要服务端.客户端两头都设置等.想想实际上用JWT会更加便捷,而且更安全和功能强大,因为除JWT的加密签名之外还可以把私密 ...

  4. EPG开发《异常排查以及解决方案》

    [框架]

  5. springmvc上传文件踩过的坑

    @RequestMapping("/addTweet") public String addTweet(TweetVO tweetVO, HttpServletRequest re ...

  6. Hamcrest的简单应用

    import org.junit.Before; import org.junit.Ignore; import org.junit.Test; import java.util.ArrayList; ...

  7. 项目中操作redis改brpop阻塞模式为订阅模式的实现-java实习笔记二

    更改项目需求以及项目之前阻塞模式问题的叙述已经在上一篇说过了,详情可参考:https://www.cnblogs.com/darope/p/10276213.html  ,https://yq.ali ...

  8. 程序员到sql笔记

    1最近准备面试,总结一下之前学过到东西.

  9. 五分钟部署一套完整的Zabbix

    Zabbix-安装与使用 安装NTP yum install -y ntp 配置ntp.conf cd /etc/ vim ntp.conf # 打开之后讲一下内容添加 server cn.ntp.o ...

  10. 字符串的排列(剑指offer)

    题目描述: 输入一个字符串,按字典序打印出该字符串中字符的所有排列.例如输入字符串abc,则打印出由字符a,b,c所能排列出来的所有字符串abc,acb,bac,bca,cab和cba. 思路: 利用 ...