问题描述  

  给定两个整数,被除数 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. Codeforces Gym101518H:No Smoking, Please(最小割)

    题目链接 题意 给出一个n*m的酒店,每个点是一个房间,要将这个酒店的房间划分成为两块(一块无烟区,一块吸烟区),相邻的两个房间之间有一条带权边,权值代表空气锁的面积,如果把这条边给去掉,那么需要花费 ...

  2. django基础知识之定义模板:

    定义模板 模板语言包括 变量 标签 { % 代码块 % } 过滤器 注释{# 代码或html #} 变量 语法: {{ variable }} 当模版引擎遇到一个变量,将计算这个变量,然后将结果输出 ...

  3. Coder必须自废的两样神功

    Coder必须自废的两样神功 大理段氏以一阳指神功驰名天下.奉六脉宝经,立天龙佛院:凭借数百载基业威名,与嵩山少林.终南全真分庭抗礼:乃宋代中华武术三大派系之一. 二指禅是一种鲜为人知的秘传功法,通过 ...

  4. 硬件设计--DC/DC电源芯片详解

    本文参考:http://www.elecfans.com/article/83/116/2018/20180207631874.html https://blog.csdn.net/wangdapao ...

  5. ASP.NET Core系列(二):创建第一个.Net Core 项目

    前面讲过 .NET Core简介及开发环境安装,本章会讲一讲ASP.NET Core 2.0的项目结构,查看完整的ASP.NET Core系列文章:https://www.cnblogs.com/zh ...

  6. 深入了解数据校验:Bean Validation 2.0(JSR380)

    每篇一句 > 吾皇一日不退役,尔等都是臣子 对Spring感兴趣可扫码加入wx群:`Java高工.架构师3群`(文末有二维码) 前言 前几篇文章在讲Spring的数据绑定的时候,多次提到过数据校 ...

  7. 小白开学Asp.Net Core 《一》

    在开篇中介绍了项目的搭建以及项目中所用到的第三方工具 本篇介绍SqlSugar和Dapper在本项目的实现 一.SqlSugar SqlSuagr的介绍就直接浏览官方网站,官网地址将在底部给出. 在本 ...

  8. 快速java环境变量配置记录

      配置java环境变量就是将java.exe和javac.exe的路径告诉系统,让系统能够找到这两个exe文件,废话不多说,直接开始如何配置环境变量,安装jdk时记住你的安装位置.(配置时必须要的) ...

  9. Python基础总结之第三天开始重新认识‘字符串’(新手可相互督促)

    年薪20万的梦想,又进了一步... 戏好多 ’字符串‘开始啦~ 字符串的定义:字符串可以用英文单引号或双引号又或者三引号包围起来. 为毛有单引号,还要有双引号和三引号??? 看案例吧: 字符串的其他使 ...

  10. jsp数据交互(二).1

    对象的作用域:   JSP中提供了四种作用域,分别是page作用域,request作用域,session作用域和application作用域. page作用域: page作用域指单一JSP页面的范围, ...