乘风破浪:LeetCode真题_029_Divide Two Integers
乘风破浪:LeetCode真题_029_Divide Two Integers
一、前言
两个整数相除,不能使用乘法除法和取余运算。那么就只能想想移位运算和加减法运算了。
二、Divide Two Integers
2.1 问题
2.2 分析与解决
通过分析,我们可以想到,如果使用加法,一次次的减下去,每减一次就加一,直到最后的减数小于除数。但是这样的时间复杂度将会是非常的大,比如100000,3这两个数,非常的耗时,那么如何加快运算呢,我们想到了移位运算。
public class Solution { /** * 题目大意: * 不使用除法,乘法和取余,求两个整数的相除的结果,如果有溢出就返回最大的整数 * * 解题思路: * 任何一个整数可以表示成以2的幂为底的一组基的线性组合, * 即num=a_0*2^0+a_1*2^1+a_2*2^2+...+a_n*2^n。 * 基于以上这个公式以及左移一位相当于乘以2,我们先让除 * 数左移直到大于被除数之前得到一个最大的基。然后接下来我们 * 每次尝试减去这个基,如果可以则结果增加加2^k,然后基继续右 * 移迭代,直到基为0为止。因为这个方法的迭代次数是按2的幂知 * 道超过结果,所以时间复杂度为O(log(n))。 * */ public int divide(int dividend, int divisor) { // 相除时溢出处理 if (divisor == 0 || dividend == Integer.MIN_VALUE && divisor == -1) { return Integer.MAX_VALUE; } // 求符号位,异或运算 int sign = ((dividend < 0) ^ (divisor < 0)) ? -1 : 1; // 求绝对值,为防止溢出使用long long dvd = Math.abs((long) dividend); long dvs = Math.abs((long) divisor); // 记录结果 int result = 0; // 被除数大于除数 while (dvd >= dvs) { // 记录除数 long tmp = dvs; // 记录商的大小 long mul = 1; while (dvd >= (tmp << 1)) { tmp <<= 1; mul <<= 1; } // 减去最接近dvd的dvs的指数倍的值(值为tmp) dvd -= tmp; // 修正结果 result += mul; } return result * sign; } }
三、总结
当我们没有办法的时候不妨想想移位操作,这样能极大的提升我们的操作速度和效率。