Table of contents
Open Table of contents
Description
Given a 0-indexed integer array nums
of size n
and two integers lower
and upper
, return the number of fair pairs.
A pair (i, j)
is fair if:
0 <= i < j < n
, andlower <= nums[i] + nums[j] <= upper
Example 1:
Input: nums = [0,1,7,4,4,5], lower = 3, upper = 6
Output: 6
Explanation: There are 6 fair pairs: (0,3), (0,4), (0,5), (1,3), (1,4), and (1,5).
Example 2:
Input: nums = [1,7,9,2,5], lower = 11, upper = 11
Output: 1
Explanation: There is a single fair pair: (2,3).
Constraints:
1 <= nums.length <= 10^5
nums.length == n
-10^9 <= nums[i] <= 10^9
-10^9 <= lower <= upper <= 10^9
Solution
Idea
First, we can note that the number of fair pairs does not change if we sort the array.
No matter index i
and j
elements are swapped or not swapped during sorting, they are still a fair pair.
- Sort the array.
- For each element in the array, we look for the first element (index
l
) greater thanupper-v
and the first element (indexr
) not less thanlower-v
. For the current element it can pair with elements in index range[l,r)
to form a fair pair, where[
(square bracket) is inclusive and)
is exclusive, i.e., range[l,r-1]
. So we increment result withr-l
. - We return the result after the iteration.
Note: it is probably easier to use C++ and Python for this idea since they have the following library functions:
- find first element/index not less than the target: C++
lower_bound()
and Pythonbisect_left()
- find first element/index greater than the target: C++
upper_bound()
and Pythonbisect_right()
Complexity: Time , Space .
The space complexity of the sorting algorithm depends on the programming language.
- In Python, the sort method sorts a list using the Timsort algorithm which is a combination of Merge Sort and Insertion Sort and has O(n) additional space.
- In Java,
Arrays.sort()
is implemented using a variant of the Quick Sort algorithm which has a space complexity of O(logn) for sorting two arrays. - In C++, the
sort()
function is implemented as a hybrid of Quick Sort, Heap Sort, and Insertion Sort, with a worse-case space complexity of O(logn).
C++
class Solution {
public:
long long countFairPairs(vector<int> &nums, int lower, int upper) {
long long res = 0;
auto beg = nums.begin(), end = nums.end();
sort(beg, end);
for (auto [i, v]: views::enumerate(nums)) // since C++23
res += upper_bound(beg + i + 1, end, upper - v)
- lower_bound(beg + i + 1, end, lower - v);
return res;
}
};
Java
There is no implementation for lower_bound()
or upper_bound()
in standard library in Java.
However, we can implement one fairly straight forward.
Since we are implementing it, it makes sense to calculate the result together with the binary search.
Because of that, this method should be faster than the C++ implementation above
(imagine instead of binary searching [beg+i+1,end]
in every iteration of the for loop,
we can use a variable to remember and reduce the initial search range in the C++ implementation).
Big O complexity is still the same.
class Solution {
public long countFairPairs(int[] nums, int lower, int upper) {
Arrays.sort(nums);
return cntLess(nums, upper + 1) - cntLess(nums, lower); // [0,upper+1)-[0,lower): [lower,upper]
}
// Calculate the number of pairs with a sum less than `value`.
private long cntLess(int[] nums, int value) {
int l = 0, r = nums.length - 1;
long res = 0;
while (l < r) {
int sum = nums[l] + nums[r];
if (sum < value) {
res += (r - l);
l++;
} else r--;
}
return res;
}
}