Alkesh

LeetCode - Divide Two Integers

Problem statement

Given two integers dividend and divisor, divide two integers without using multiplication, division, and mod operator.

The integer division should truncate toward zero, which means losing its fractional part. For example, 8.345 would be truncated to 8, and -2.7335 would be truncated to -2.

Return the quotient after dividing dividend by divisor.

Note: Assume we are dealing with an environment that could only store integers within the 32-bit signed integer range: [−2^31, 2^31 − 1]. For this problem, if the quotient is strictly greater than 2^31 - 1, then return 2^31 - 1, and if the quotient is strictly less than -2^31, then return -2^31.

Problem statement taken from: https://leetcode.com/problems/divide-two-integers

Example 1:

Input: dividend = 10, divisor = 3
Output: 3
Explanation: 10/3 = 3.33333.. which is truncated to 3.

Example 2:

Input: dividend = 7, divisor = -3
Output: -2
Explanation: 7/-3 = -2.33333.. which is truncated to -2.

Constraints:

- -2^31 <= dividend, divisor <= 2^31 - 1
- divisor != 0

Explanation

Brute Force

We are not allowed to use multiplication, division, and mod operator. We can make use of the subtraction operator to divide two integers and find the answer. We keep subtracting the divisor from the dividend till the time dividend is greater than or equal to the divisor. We keep a count of the number of subtractions. This count is equal to the quotient of the two numbers.

We must also keep track of the sign of the dividend and the divisor. A C++ snippet of this approach is as follows:

int divide(int dividend, int divisor) {
    if (divisor == 1)
        return dividend;

    if (dividend == INT_MIN) {
        dividend++;
    }

    int sign = -1, quotient = 0;

    if ((dividend < 0 && divisor < 0) || (dividend >= 0 && divisor >= 0))
        sign = 1;

    dividend = abs(dividend);
    divisor = abs(divisor);

    while (dividend >= divisor) {
        dividend -= divisor;
        quotient++;
    }

    return quotient * sign;
}

The time complexity of this approach is O(abs(m - n) / n), where m is the dividend and n is the divisor. The space complexity of this approach is O(1). We are not using any extra space to store the numbers, the space complexity of this solution is constant, i.e. O(1).

Binary Search

We can optimize the above approach using Binary Search. We know for a particular dividend and divisor, quotient * divisor <= dividend. For a number a, which is less than the quotient this equation always holds true a * divisor <= dividend. Similarly, for a number b which is greater than the quotient, the equation b * divisor >= dividend is true.

We know the quotient will lie between 0 and the dividend. We perform a binary search on this given range 0 to the dividend. If mid * divisor > dividend, we search the left side and set high = mid - 1. Else we search the right side and set low = mid + 1. The value of mid that satisfies the condition will be our quotient.

A C++ snippet of this approach is as follows:

int divide(int dividend, int divisor) {
    unsigned long long int low = 0;
    unsigned long long int high = abs(dividend);

    if (dividend == INT_MIN) {
        if (divisor == -1) {
            return INT_MAX;
        } else if (divisor == 1) {
            return INT_MIN;
        }
    }

    int quotient = 0;
    while (low <= high) {
        unsigned long long int mid = (low + high) / 2;
        if (abs(divisor) * mid <= abs(dividend)) {
            quotient = mid;
            low = mid + 1;
        } else {
            high = mid - 1;
        }
    }

    return ((dividend < 0 && divisor < 0) || (dividend >= 0 && divisor >= 0) ? quotient : -quotient);
}

The time complexity of this approach is O(log n), where n is the dividend. The space complexity of this approach is O(1).

Bit Manipulation

We find the greatest power of 2 when multiplied by the quotient gives us a quotient that is just smaller than the dividend. After finding this power, we add it to our answer. We then reduce the dividend by the divisor multiplied by this power of 2.

Let's check the algorithm to understand it clearly.

Algorithm

- if dividend == divisor
    return 1

- set isPositive = (dividend < 0 == divisor < 0)
      absDividend = abs(dividend)
      absDivisor = abs(divisor)
      answer = 0

- loop while absDividend >= absDivisor
    int quotient = 0

    loop while absDividend > (absDivisor << (quotient + 1))
      quotient++
    while end

    update answer = answer + (1 << quotient)

    update absDividend = absDividend - (absDivisor << quotient)
- while end

- if answer == (1 << 31) and isPositive
    return INT_MAX

- return isPositive ? answer : -answer

The time complexity of this approach is O(log n), where n is the dividend. The space complexity of this approach is O(1).

Let's check out our solutions in C++, Golang, and Javascript.

C++ solution

class Solution {
public:
    int divide(int dividend, int divisor) {
        if (dividend == divisor) {
            return 1;
        }

        bool isPositive = (dividend < 0 == divisor < 0);
        unsigned int absDividend = abs(dividend);
        unsigned int absDivisor = abs(divisor);
        unsigned int answer = 0;

        while (absDividend >= absDivisor) {
            int quotient = 0;
            while (absDividend > (absDivisor << (quotient + 1)))
                quotient++;

            answer += (1 << quotient);

            absDividend -= (absDivisor << quotient);
        }

        if (answer == (1 << 31) and isPositive)
            return INT_MAX;

        return isPositive ? answer : -answer;
    }
};

Golang solution

func abs(a int) int {
    if a < 0 {
        return a * -1
    }

    return a
}

func divide(dividend int, divisor int) int {
    if dividend == divisor {
        return 1
    }

    isPositive := true

    if (dividend < 0 && divisor > 0) || (dividend > 0 && divisor < 0) {
        isPositive = false
    }

    absDividend := abs(dividend)
    absDivisor := abs(divisor)
    answer := 0

    for absDividend >= absDivisor {
        quotient := 0

        for absDividend > (absDivisor << (quotient + 1)) {
            quotient++
        }

        answer += (1 << quotient)

        absDividend -= (absDivisor << quotient)
    }

    if answer == (1 << 31) && isPositive {
        return math.MaxInt32
    }

    if isPositive {
        return answer
    } else {
        return -answer
    }
}

JavaScript solution

var divide = function(dividend, divisor) {
    if (dividend == divisor) {
        return 1;
    }

    let isPositive = (dividend < 0 == divisor < 0);
    let absDividend = Math.abs(dividend);
    let absDivisor = Math.abs(divisor);
    let answer = 0;

    while (absDividend >= absDivisor) {
        let quotient = 0;
        while (absDividend > (absDivisor << (quotient + 1)))
            quotient++;

        answer += (1 << quotient);

        absDividend -= (absDivisor << quotient);
    }

    if (answer == (1 << 31) && isPositive)
        return Number.MAX_SAFE_INTEGER;

    return isPositive ? answer : -answer;
};

Dry Run

Let's dry-run our algorithm for a few examples to see how the solution works.

Input dividend = 10
      divisor = 3

Step 1: if dividend == divisor
           10 == 3
           false

Step 2: isPositive = (dividend < 0 == divisor < 0)
                   = (10 < 0 == 3 < 0)
                   = (false == false)
                   = true
        absDividend = abs(dividend)
                    = abs(10)
                    = 10
        absDivisor = abs(absDivisor)
                    = abs(3)
                    = 3
        answer = 0

Step 3: loop while absDividend >= absDivisor
          10 >= 3
          true

          quotient = 0

          loop while absDividend > (absDivisor << (quotient + 1))
            10 > (3 << (0 + 1))
            10 > (3 << 1)
            10 > 6
            true

            quotient++
            quotient = quotient + 1
                     = 1

          loop while absDividend > (absDivisor << (quotient + 1))
            10 > (3 << (1 + 1))
            10 > (3 << 2)
            10 > 12
            false

          answer += (1 << quotient)
          answer = answer + (1 << quotient)
                 = 0 + 1 << 1
                 = 0 + 2
                 = 2

          absDividend -= (absDivisor << quotient)
          absDividend = absDividend - (absDivisor << quotient)
                      = 10 - (3 << 1)
                      = 10 - 6
                      = 4

Step 4: loop while absDividend >= absDivisor
          4 >= 3
          true

          quotient = 0

          loop while absDividend > (absDivisor << (quotient + 1))
            4 > (3 << (0 + 1))
            4 > (3 << 1)
            4 > 6
            false

          answer += (1 << quotient)
          answer = answer + (1 << quotient)
                 = 2 + 1 << 0
                 = 2 + 1
                 = 3

          absDividend -= (absDivisor << quotient)
          absDividend = absDividend - (absDivisor << quotient)
                      = 4 - (3 << 0)
                      = 4 - 3
                      = 1

Step 5: loop while absDividend >= absDivisor
          1 >= 3
          false

Step 6: if answer == (1 << 31) && isPositive
           3 == 2147483648 && true
           false && true
           false

Step 7: return isPositive ? answer : -answer
            true ? 3 : -3

We return the answer as 3.
Share this post!