给定两个整数,被除数 dividend 和除数 divisor。将两数相除,要求不使用乘法、除法和 mod 运算符。
返回被除数 dividend 除以除数 divisor 得到的商。
整数除法的结果应当截去(truncate)其小数部分,例如:truncate(8.345) = 8 以及 truncate(-2.7335) = -2
示例 1:
输入: dividend = 10, divisor = 3
输出: 3
解释: 10/3 = truncate(3.33333…) = truncate(3) = 3
示例 2:
输入: dividend = 7, divisor = -3
输出: -2
解释: 7/-3 = truncate(-2.33333…) = -2
提示:
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/divide-two-integers
解题思路
这题本质就是考的二分搜索,但是如何二分搜索是最大的问题。
很多人用的二进制移位操作,这明显不符合题意,
就算二进制左移1位,乘以2符合,可以用加法代替,那右移除以二怎么符合?
不要说什么计算机二进制很快,底层计算机就是二进制,你在麻痹自己。
这题本意不是考察二进制,而是考察二进制搜索,而且我认为这是一个非常好的题目。
我们理解的二分搜索其本质思想是二进制搜索,在这题中得以体现出来。
这个题首先需要考虑正负号,处理为分子分母全是正数,
其次在返回的时候要注意是否溢出,如果溢出要判断。
核心是div函数怎么写?例如方法1中的div函数,
利用二进制搜索的思想就是,
每次利用加法,将当前的divisor乘以两倍,并同时用multiple记录下乘以了2的多少次方,
multiple的变化过程是1,2,4,8,16 。。。
因为任何一个数都可以用二进制的方法得到,所以我们可以利用二进制的思想来代表乘数multiple,
最终能够得到一个divisor * multiple = dividend的multiple。
代码
# 方法1:递归
class Solution:
def divide(self, dividend: int, divisor: int) -> int:
MIN_INT, MAX_INT = -2147483648, 2147483647 # [−2**31, 2**31−1]
flag = 1 # 存储正负号,并将分子分母转化为正数
if dividend < 0: flag, dividend = -flag, -dividend
if divisor < 0: flag, divisor = -flag, -divisor
def div(dividend, divisor): # 例:1023 / 1 = 512 + 256 + 128 + 64 + 32 + 16 + 8 + 4 + 1
if dividend < divisor:
return 0
cur = divisor
multiple = 1
while cur + cur < dividend: # 用加法求出保证divisor * multiple <= dividend的最大multiple
cur += cur # 即cur分别乘以1, 2, 4, 8, 16...2^n,即二进制搜索
multiple += multiple
return multiple + div(dividend - cur, divisor)
res = div(dividend, divisor)
res = res if flag > 0 else -res # 恢复正负号
if res < MIN_INT: # 根据是否溢出返回结果
return MIN_INT
elif MIN_INT <= res <= MAX_INT:
return res
else:
return MAX_INT
# 方法2:迭代
class Solution:
def divide(self, dividend: int, divisor: int) -> int:
MIN_INT, MAX_INT = -2147483648, 2147483647 # [−2**31, 2**31−1]
flag = 1 # 存储正负号,并将分子分母转化为正数
if dividend < 0: flag, dividend = -flag, -dividend
if divisor < 0: flag, divisor = -flag, -divisor
res = 0
while dividend >= divisor: # 例:1023 / 1 = 512 + 256 + 128 + 64 + 32 + 16 + 8 + 4 + 1
cur = divisor
multiple = 1
while cur + cur < dividend: # 用加法求出保证divisor * multiple <= dividend的最大multiple
cur += cur # 即cur分别乘以1, 2, 4, 8, 16...2^n,即二进制搜索
multiple += multiple
dividend -= cur
res += multiple
res = res if flag > 0 else -res # 恢复正负号
if res < MIN_INT: # 根据是否溢出返回结果
return MIN_INT
elif MIN_INT <= res <= MAX_INT:
return res
else:
return MAX_INT