LCR 001. 两数相除
给定两个整数 a
和 b
,求它们的除法的商 a/b
,要求不得使用乘号 '*'
、除号 '/'
以及求余符号 '%'
。
注意:
整数除法的结果应当截去(
truncate
)其小数部分,例如:truncate(8.345) = 8
以及truncate(-2.7335) = -2
假设我们的环境只能存储 32 位有符号整数,其数值范围是
[−231, 231−1]
。本题中,如果除法结果溢出,则返回231 − 1
示例 1:
输入:a = 15, b = 2
输出:7
解释:15/2 = truncate(7.5) = 7
示例 2:
输入:a = 7, b = -3
输出:-2
解释:7/-3 = truncate(-2.33333..) = -2
示例 3:
输入:a = 0, b = 1
输出:0
示例 4:
输入:a = 1, b = 1
输出:1
提示:
-231 <= a, b <= 231 - 1
b != 0
思路
俄罗斯农民算法(移位相加法)+倍数相减法
二分搜索(参考答案)
首先这种不能使用xxx方法的题目,一般机试时是不会出的,因为限制不了编辑器,人工阅卷又需要耽误时间。
但正因为不会出现在机试中,所以在手撕算法时却会成为热门,所以请不要违背题意的直接使用乘、除提交题目。
拿示例一来说,一个最直观的方法就是循环使用a-b,计算减了多少次,即可得到结果。
但请注意a、b的取值范围,如果a为2 ^ 31−1,b为1那么仅一次就需要计算2 ^ 31−1次,所以这个思路是不可行的。
那么,既然不能从被除数上下手,能否通过除数进行操作呢?如果大家对二分搜索熟悉,相信很快能找到思路。
初始化返回值ret = 0
如果被除数大于除数,则除数扩大一倍
若被除数仍大于除数,这除数再次扩大一倍
直到除数下一次翻倍比被除数大时,将被除数减去除数,并将ret+=除数扩大的倍数,结束这一轮循环
重复2、3、4,直到被除数小于除数,终止循环并返回ret即可。
另外:在计算开始时,我们需要注意判断除数和被除数的正负号问题。如果是Python,只需要记录正负号即可。但对于Java而言,
由于负数的取值比正数大1,转换成正数计算会造成溢出,所以简单的方式是将数字转化为负数来进行比较。
作者:清风Python
链接:https://leetcode.cn/problems/xoh6Oh/solutions/940024/shua-chuan-jian-zhi-offer-day01-zheng-sh-8u0s/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
代码
俄罗斯农民算法(移位相加法)+倍数相减法
class Solution {
public int divide(int a, int b) {
boolean flag = (a < 0) ^ (b < 0);
if (a == Integer.MIN_VALUE) {
if (b == 1) {
return Integer.MIN_VALUE;
} else if (b == -1) {
return Integer.MAX_VALUE;
}
}
if (a > 0) {
a = -a;
}
if (b > 0) {
b = -b;
}
if (a > b) {
return 0;
}
int mid = 1;
int temp = 1;
while (temp > 0) {
int product = shiftAdd(mid, b);
if (product == a) {
return flag ? -mid : mid;
} else if (a - product <= product) {
temp = mid;
mid <<= 1;
} else if (a - product <= shiftAdd(temp, b)) {
mid += temp;
temp >>= 1;
} else {
temp >>= 1;
}
}
return flag ? -mid : mid;
}
private int shiftAdd(int a, int b) {
int result = 0;
while (a > 0) {
if ((a & 1) == 1) {
result += b;
}
a >>= 1;
b <<= 1;
}
return result;
}
}
二分搜索
class Solution {
public int divide(int a, int b) {
boolean flag = (a < 0) ^ (b < 0);
if (a > 0) {
a = -a;
}
if (b > 0) {
b = -b;
}
int ret = calc(a, b);
if (!flag && ret == Integer.MIN_VALUE) {
ret++;
}
return flag ? ret : -ret;
}
private int calc(int a, int b) {
int ret = 0;
while (a <= b) {
int cnt = 1;
int val = b;
while (val >= Integer.MIN_VALUE >> 1 && a <= val << 1) {
cnt += cnt;
val += val;
}
ret -= cnt;
a -= val;
}
return ret;
}
}