Leetcode Pens and Pencils problem in O(log(n)) time

Published on

There is a leetcode problem Number of Ways to Buy Pens and Pencils. It has the following requirement:

You are given an integer total indicating the amount of money you have. You are also given two integers cost1 and cost2 indicating the price of a pen and pencil respectively. You can spend part or all of your money to buy multiple quantities (or none) of each kind of writing utensil.

Return the number of distinct ways you can buy some number of pens and pencils.

 

Example 1:

Input: total = 20, cost1 = 10, cost2 = 5
Output: 9
Explanation: The price of a pen is 10 and the price of a pencil is 5.
- If you buy 0 pens, you can buy 0, 1, 2, 3, or 4 pencils.
- If you buy 1 pen, you can buy 0, 1, or 2 pencils.
- If you buy 2 pens, you cannot buy any pencils.
The total number of ways to buy pens and pencils is 5 + 3 + 1 = 9.

Example 2:

Input: total = 5, cost1 = 10, cost2 = 10
Output: 1
Explanation: The price of both pens and pencils are 10, which cost more than total, so you cannot buy any writing utensils. Therefore, there is only 1 way: buy 0 pens and 0 pencils.

The naive solution

The naive solution will look like this: long long result = 0; while (total >= 0) { result += total / cost2 + 1; total -= cost1; } return result;

It calculates for each number of item 1 the possible number of item 2 and sums the results. The worst case execution time would be O(total/max(cost1,cost2)) if we optimize by subtracting the greatest of cost1 and cost2 from total, but it is still linear (proportional to total) in the worst case.

Can we do faster? It turns out yes.

The problem is equivalent to a geometrical problem of finding the number of discrete points that satisfy these conditions:

For 11x + 17y ≤ 40 the graphical representation will be like this:

Triangle 1

The red triangle contains 8 points (including the ones that touch its sides) that have integer (whole) coordinates.

So how can we quickly find the number of such points in any triangle?

A better approach

We can calculate the total amount of points by taking the advantage of the fact that we can calculate the number of the points in a triangle with legs a and k*a in a constant amount of time, where k is a whole number and a is positive number (can be non whole number too).

Let's consider a triangle with k=2:

Triangle 2

We can easily calculate the number of points within a constant amount of time with the help of arithmetic progression formula (S = (a1 + an) * n / 2), since every next (upper) row has k (in this example 2) less points than the previous one and we can calculate the number of elements and the points in the bottom row in a constant amount of time by integer division / rounding. I.e. it's 6 + 4 + 2 = (6 + 2) * 3 / 2 = 12 in this case.

OK, but not all cases will be triangles with legs of a and a*k, so how can we count for any triangle? Yes, not all triangles are like this, but we can extract such triangle from any triangle and transform the remaining figure to a smaller triangle and repeat this process until the remaining figure is small enough to calculate its points in a constant amount of time.

This is how it works

Triangle 3

Since in each iteration one of the sides of the next triangle decreases by a factor of at least 2 (similar to Euclidean algorithm), the complexity will be logarithmical in the worst case.

When we have cost1, cost2, cost1 ≤ cost2 (the horizontal leg shouldn't be smaller than the vertical leg, otherwise we swap cost1 and cost2) and total, the figures will have such properties:

  1. The pink triangle that satisfies the condition cost1 * x + cost2 * y ≤ total will have width (horizontal leg length) w = total / cost1 and height (vertical leg length) h = total / cost2.
  2. k = floor(w / h) = floor((total / cost1) / (total / cost2)) = floor(cost2 / cost1).
  3. The integer part of the height will be n = floor(h) = floor(total / cost2). The number of integer coordinates in the vertical leg will be n + 1.
  4. The number of integer coordinates in the base of the gray triangle will be l = floor(k * h) + 1 = floor(k * total / cost2) + 1.
  5. The number of points with integer coordinates in the gray triangle will be a sum of an arithmetic progression: S = (a1 + an+1) * (n + 1) / 2 = (l + l - k * n) * (n + 1) / 2 = (n + 1) * (2 * l - k * n) / 2.
  6. The X coordinate of the intersection of the blue line and X axis will be floor(k * h) + 1 = l.
  7. The Y coordinate of the intersection of the blue line (this is also the length of the vertical leg of the blue triangle) and the hypotenuse of the pink triangle after solving the linear equation will be (total - cost1 * l) / (cost2 - cost1 * k) (for now ignore the case where the denominator is 0).
  8. The horizontal leg of the blue triangle (also the base of the red triangle, since the blue triangle was the result of the red triangle horizontal skewing) will be w - l = total / cost1 - l = (total - cost1 * l) / cost1.
  9. Just notice that for both horizontal and vertical legs' lengths of the blue triangle the numerator is total - cost1 * l. This means that the blue triangle can be considered as a smaller version of the original problem with this parameters:
    • totalnew = total - cost1 * l
    • cost1new = cost2 - k * cost1
    • cost2new = cost1
    This trick also eliminates the need for division. So it will not cause any problems even when some of the denominators are 0. We will always check if total is large enough proportionally to cost2 in order to continue to compute the result recursively or just fall back to the "naive" approach. totalnew can even be negative, but the algorithm will still work correctly. It might be not that obvious but it is true.

Complexity

Code (C++)

class Solution { public: long long waysToBuyPensPencils(int total, int cost1, int cost2) { if (cost2 < cost1) { swap(cost1, cost2); } long long res = 0; while (true) { if (4 * cost2 > total) { while (total >= 0) { res += total / cost1 + 1; total -= cost2; } return res; } auto k = cost2 / cost1; auto n = total / cost2; auto l = k * total / cost2 + 1; res += (long long)(n + 1) * (2 * l - k * n) / 2; total -= cost1 * l; auto temp = cost2; cost2 = cost1; cost1 = temp - k * cost1; } } };

I also posted this solution here.