【每天学习一点算法 2026/04/13】两数相除

每天学习一点算法 2026/04/13

题目:两数相除

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

整数除法应该向零截断,也就是截去(truncate)其小数部分。例如,8.345 将被截断为 8 ,-2.7335 将被截断至 -2 。

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

注意:假设我们的环境只能存储 32 位 有符号整数,其数值范围是 −231, 231 − 1 。本题中,如果商 严格大于 231 − 1 ,则返回 231 − 1 ;如果商 严格小于 -231 ,则返回 -231 。

这道题可给我折磨坏了,其实最简单的办法就是用 dividend 一直减 divisor ,直到结果小于 divisor,减的次数就是我们要求的结果。

typescript 复制代码
function divide(dividend: number, divisor: number): number {
    let res = 0
    while (dividend >= divisor) {
      dividend -= divisor
      res++
    }
    return res
};

但是不用想肯定是超时的,这个替换一个思路其实是要在 [1, dividend] 的范围内找到一个数字 n,使他的 dividend - divisor < divisor × n <= dividend

这样我们就是可以利用二分查找,减少复杂度,但是求倍数又是一个问题了,官方建议用使用 快速乘 类似 Pow(x, n) 的 快速幂,像这样:

typescript 复制代码
function helper (x: number, n: number) {
  // 较小的数作为倍数
  if (x < n) {
    [x, n] = [n, x]
  }
  if (n === 1) {
    return x
  }
  if (x === 1) {
    return n
  }
  const m = n >> 1 // = Math.floor(n / 2) 利用右移代替除二
  if (m + m === n) {
      // 偶数
    return helper(x, m) + helper(x, m)
  } else {
    // 奇数
    return helper(x, m) + helper(x, m) + x
  }
}

但是遇到 dividend = 1026117192 divisor = -874002063 这种还是会超时,欺骗我感情。

最后呢,我们还是利用累加来判断 x × n 是否大于 dividend 来实现

typescript 复制代码
function divide(dividend: number, divisor: number): number {
    if (dividend === 0) return 0
    const MAX = 2147483647
    const MIN = -2147483648
    let res = 0
    let flag = (divisor > 0 && dividend > 0) || (divisor < 0 && dividend < 0) // 正负标识
    // 取绝对值做计算可以减少很多条件判断
    dividend = Math.abs(dividend) 
    divisor = Math.abs(divisor)
    if (dividend < divisor) return 0 // 除数大于被除数直接返回 0
    if (divisor === 1) {
      // 除数是 1 结果直接等于被除数
      res = dividend
    }
    if (dividend === divisor) {
      // 被除数和除数相等直接结果为1
      res = 1
    }
    
    // 辅助函数用于判断 x * n 是否大于 dividend
    function helper (x: number, n: number) {
      // 较小的数作为倍数
      if (x < n) {
        [x, n] = [n, x]
      }
      let sum = 0
      // 累加判断结果是否大于 dividend
      while (n > 0) {
        sum += x
        if (sum > dividend) {
          return false
        }
        n--
      }
      return true
    }
    
    // 二分查找目标倍数
    let left = 1, right = dividend
    while (left <= right && divisor > 1 && dividend !== divisor) {
      const mid = left + ((right - left) >> 1) // 利用右移代替除二
      const chenji = helper(divisor, mid)
      if (!chenji) {
        // divisor的mid倍大于dividend
        right = mid - 1
      } else {
        // divisor的mid倍小于等于dividend
        left = mid + 1
        res = mid
      }
    }
    // 最后根据符号返回对应结果,注意边界判断
    if (flag) {
      return Math.min(MAX, res)
    } else {
      return Math.max(MIN, -res)
    }
};

题目来源:力扣(LeetCode)

相关推荐
JieE21214 小时前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack201 天前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树1 天前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
JieE2122 天前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE2122 天前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
vivo互联网技术2 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦2 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
用户497863050732 天前
(一)小红的数组操作
算法·编程语言
怕浪猫2 天前
Electron 系列文章封面图
算法·架构·前端框架