• ☆打卡算法☆LeetCode 29、两数相除 算法解析


    推荐阅读

    大家好,我是小魔龙,Unity3D软件工程师,VR、AR,虚拟仿真方向,不定时更新软件开发技巧,生活感悟,觉得有用记得一键三连哦。

    一、题目

    1、算法题目

    “给定两个整数,进行相除,不能使用乘法、除法和mod运算符。”

    题目链接:

    来源:力扣(LeetCode)

    链接:29. 两数相除 - 力扣(LeetCode) (leetcode-cn.com)

    2、题目描述

    给定两个整数,被除数 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
    
    • 1
    • 2
    • 3
    • 4
    示例 2:
    输入: dividend = 7, divisor = -3
    输出: -2
    解释: 7/-3 = truncate(-2.33333..) = -2
    
    • 1
    • 2
    • 3
    • 4

    二、解题

    1、思路分析

    这个题首先需要考虑整数溢出或边界情况:

    当被除数为最小值-232时:

    • 除数为1,可以直接返回答案-232
    • 除数为-1,那么答案为232,产生了溢出,需要返回232-1

    当除数为最小值-232时:

    • 除数同样为-232,返回答案1
    • 其他情况,返回0

    当除数为0时,返回0。

    然后,记被除数和除数为X和Y,结果为Z,使用二分查找法,得到X/Y的最大结果Z,即使得Z x Y ≥ Z成立。

    Z x Y的值,可以使用快速乘方法得到。

    2、代码实现

    代码参考:

    public class Solution {
        public int Divide(int dividend, int divisor) {
            // 考虑被除数为最小值的情况
            if (dividend == int.MinValue) {
                if (divisor == 1) {
                    return int.MinValue;
                }
                if (divisor == -1) {
                    return int.MaxValue;
                }
            }
            // 考虑除数为最小值的情况
            if (divisor == int.MinValue) {
                return dividend == int.MinValue ? 1 : 0;
            }
            // 考虑被除数为 0 的情况
            if (dividend == 0) {
                return 0;
            }
            
            // 一般情况,使用二分查找
            // 将所有的正数取相反数,这样就只需要考虑一种情况
            bool rev = false;
            if (dividend > 0) {
                dividend = -dividend;
                rev = !rev;
            }
            if (divisor > 0) {
                divisor = -divisor;
                rev = !rev;
            }
            
            int left = 1, right = int.MaxValue, ans = 0;
            while (left <= right) {
                // 注意溢出,并且不能使用除法
                int mid = left + ((right - left) >> 1);
                bool check = quickAdd(divisor, mid, dividend);
                if (check) {
                    ans = mid;
                    // 注意溢出
                    if (mid == int.MaxValue) {
                        break;
                    }
                    left = mid + 1;
                } else {
                    right = mid - 1;
                }
            }
    
            return rev ? -ans : ans;
        }
    
        // 快速乘
        public bool quickAdd(int y, int z, int x) {
            // x 和 y 是负数,z 是正数
            // 需要判断 z * y >= x 是否成立
            int result = 0, add = y;
            while (z != 0) {
                if ((z & 1) != 0) {
                    // 需要保证 result + add >= x
                    if (result < x - add) {
                        return false;
                    }
                    result += add;
                }
                if (z != 1) {
                    // 需要保证 add + add >= x
                    if (add < x - add) {
                        return false;
                    }
                    add += add;
                }
                // 不能使用除法
                z >>= 1;
            }
            return true;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-yS2SVtG4-1658999020983)(https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/98550cc2f36440619dd710bc71326cbe~tplv-k3u1fbpfcp-watermark.image?)]

    3、时间复杂度

    时间复杂度 : O(log2 C)

    其中 C 表示 32 位整数的范围。二分查找的次数为 O(logC),其中的每一步我们都需要 O(logC) 使用「快速乘」算法判断 Z×Y≥X 是否成立,因此总时间复杂度为 O(log2 C)

    空间复杂度: O(1)

    只用到常数级的变量。

    三、总结

    如果我们将被除数和除数的其中(恰好)一个变为了正数,那么在返回答案之前,我们需要对答案也取相反数。

  • 相关阅读:
    程序员应该专注技术还是转管理?
    STM32F407ZGT6移植HC-05蓝牙模块
    Flutter 必须知道的布局规则
    服务器虚拟化有什么好处
    javaSE复习
    NNDL 实验八 网络优化与正则化(3)不同优化算法比较
    C#基础总结三
    性能优化-如何爽玩多线程来开发
    【双指针】滑动窗口经典例题 力扣
    hadoop生态圈之hive面试(一)
  • 原文地址:https://blog.csdn.net/q764424567/article/details/126039199