29. 两数相除
- 2019 年 10 月 7 日
- 筆記
题目描述
给定两个整数,被除数 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。
来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/divide-two-integers 著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
思路
符合直觉的做法是,减数一次一次减去被减数,不断更新差,直到差小于0,我们减了多少次,结果就是多少。
核心代码:
let acc = divisor; let count = 0; while (dividend - acc >= 0) { acc += divisor; count++; } return count;
这种做法简单直观,但是性能却比较差. 下面来介绍一种性能更好的方法。

通过上面这样的分析,我们直到可以使用二分法来解决,性能有很大的提升。
关键点解析
- 二分查找
- 正负数的判断中,这样判断更简单。
const isNegative = dividend > 0 !== divisor > 0;
代码
/* * @lc app=leetcode id=29 lang=javascript * * [29] Divide Two Integers */ /** * @param {number} dividend * @param {number} divisor * @return {number} */ var divide = function(dividend, divisor) { if (divisor === 1) return dividend; // 这种方法很巧妙,即符号相同则为正,不同则为负 const isNegative = dividend > 0 !== divisor > 0; const MAX_INTERGER = Math.pow(2, 31); const res = helper(Math.abs(dividend), Math.abs(divisor)); // overflow if (res > MAX_INTERGER - 1 || res < -1 * MAX_INTERGER) { return MAX_INTERGER - 1; } return isNegative ? -1 * res : res; }; function helper(dividend, divisor) { // 二分法 if (dividend <= 0) return 0; if (dividend < divisor) return 0; if (divisor === 1) return dividend; let acc = 2 * divisor; let count = 1; while (dividend - acc > 0) { acc += acc; count += count; } // 直接使用位移运算,比如acc >> 1会有问题 const last = dividend - Math.floor(acc / 2); return count + helper(last, divisor); }
相关题目
- 875.koko-eating-bananas