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