29.两数相除

给定两个整数,被除数 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。

//效率太低超出时间限制
class Solution {
    public int divide(int dividend, int divisor) {
        int n = 0;
        if(dividend > 0 && divisor > 0) {
            while(dividend >= divisor) {
                dividend -= divisor;
                n++;
            }
            if(n < Integer.MAX_VALUE && n > Integer.MIN_VALUE) {
                return n;
            }
            return Integer.MAX_VALUE;
        }else if((dividend > 0 && divisor < 0) || (dividend < 0 && divisor > 0)) {
            dividend = Math.abs(dividend);
            divisor = Math.abs(divisor);
            while(dividend >= divisor) {
                dividend -= divisor;
                n++;
            } 
            if(n < Integer.MAX_VALUE && n > Integer.MIN_VALUE) {
                return -n;
            }
            return Integer.MAX_VALUE;
        } else if(dividend < 0 && divisor < 0){
            while(dividend <= divisor) {
                dividend -= divisor;
                n++;
            }  
            if(n < Integer.MAX_VALUE && n > Integer.MIN_VALUE) {
                return n;
            }
            return Integer.MAX_VALUE;
        } else {
            return 0;
        }
        
        
    }
}
//位移法
/**
     * 我们可以把一个dividend(被除数)先除以2^n,n最初为31,不断减小n去试探,当某个n满足dividend/2^n>=divisor时,
     *
     * 表示我们找到了一个足够大的数,这个数*divisor是不大于dividend的,所以我们就可以减去2^n个divisor,以此类推
     *
     * 我们可以以100/3为例
     *
     * 2^n是1,2,4,8...2^31这种数,当n为31时,这个数特别大,100/2^n是一个很小的数,肯定是小于3的,所以循环下来,
     *
     * 当n=5时,100/32=3, 刚好是大于等于3的,这时我们将100-32*3=4,也就是减去了32个3,接下来我们再处理4,同样手法可以再减去一个3
     *
     * 所以一共是减去了33个3,所以商就是33
     */
class Solution {
    public int divide(int dividend, int divisor) {
        if (dividend == 0) {
            return 0;
        }
        if (dividend == Integer.MIN_VALUE && divisor == -1) {
            return Integer.MAX_VALUE;
        }
        boolean negative;
        negative = (dividend ^ divisor) <0;//用异或来计算是否符号相异
        long t = Math.abs((long) dividend);
        long d= Math.abs((long) divisor);
        int result = 0;
        for (int i=31; i>=0;i--) {
            if ((t>>i)>=d) {//找出足够大的数2^n*divisor
                result+=1<<i;//将结果加上2^n
                t-=d<<i;//将被除数减去2^n*divisor
            }
        }
        return negative ? -result : result;//符号相异取反
    }
}
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 母亲的饼折 说起饼折,一直以为是我家乡独有的一种吃食,至于这一称谓应该是哪两个字却未曾考证过。想起摊饼折最后起锅时...
    逆水漂流阅读 3,077评论 0 0
  • 简评:之前用的可是 3.2,早就该升级了啊。 Github 的 Rails 升级花了大约一年半的时间,这是有原因的...
    极小光阅读 4,346评论 0 2
  • 认识你真好, 总能带来阳光; 认识你真好, 因为你有正能量; 认识你真好, 虽然你不在我身边,却一直在心间。 认识...
  • 老师常忽悠孩子要立志当科学家、军事家、作家。其实孩子长大后才明白这个家,那个家都不如追求身体自由、时间自由、财务自...
    沐芝阳阅读 897评论 0 0
  • 夜晚,宁静而幽暗。 没有蝉鸣蛙声,没有鸡鸣狗吠。 房间里, 床头亮着微弱的光, 照着稚嫩的脸庞, 惹得黑曜石般的眸...
    胖子萌阅读 1,046评论 0 2