LeetCode Meditations: Missing Number
Let's start with the description for this problem:
Given an array
nums
containingn
distinct numbers in the range[0, n]
, return the only number in the range that is missing from the array.
For example:
Input: nums = [3, 0, 1]
Output: 2
Explanation: n = 3 since there are 3 numbers, so all numbers are in the range [0, 3]. 2 is the missing number in the range since it does not appear in nums.
Input: nums = [3, 0, 1]
Output: 2
Explanation: n = 3 since there are 3 numbers, so all numbers are in the range [0, 3]. 2 is the missing number in the range since it does not appear in nums.
Or:
Input: nums = [0, 1]
Output: 2
Explanation: n = 2 since there are 2 numbers, so all numbers are in the range [0, 2]. 2 is the missing number in the range since it does not appear in nums.
Input: nums = [0, 1]
Output: 2
Explanation: n = 2 since there are 2 numbers, so all numbers are in the range [0, 2]. 2 is the missing number in the range since it does not appear in nums.
Or:
Input: nums = [9, 6, 4, 2, 3, 5, 7, 0, 1]
Output: 8
Explanation: n = 9 since there are 9 numbers, so all numbers are in the range [0, 9]. 8 is the missing number in the range since it does not appear in nums.
Input: nums = [9, 6, 4, 2, 3, 5, 7, 0, 1]
Output: 8
Explanation: n = 9 since there are 9 numbers, so all numbers are in the range [0, 9]. 8 is the missing number in the range since it does not appear in nums.
It's also stated that all the numbers of nums
are unique.
One easy way to solve this is to get the total sum of the range, then subtract the sum of the given array. What is left will be the missing number.
It can be done using reduce
to sum the numbers, like this:
function missingNumber(nums: number[]): number {
return Array.from({
length: nums.length + 1
}, (_, idx) => idx).reduce((acc, item) => acc + item, 0)
- nums.reduce((acc, item) => acc + item, 0);
}
function missingNumber(nums: number[]): number {
return Array.from({
length: nums.length + 1
}, (_, idx) => idx).reduce((acc, item) => acc + item, 0)
- nums.reduce((acc, item) => acc + item, 0);
}
First, we create an array with values from 0
to nums.length + 1
and get its sum, then subtract the sum of nums
from it.
However, the time and space complexities will be with this solution as we create an array for the range.
We can have a more (storage-wise) efficient solution using bit manipulation. In fact, we can use an XOR operation to help us with that.
To remember, XOR results in 1 if both of the bits are different — that is, one of them is 0 and the other is 1. When we XOR a number with itself, it will result in 0, as all the bits are the same.
For example, 3 in binary is 11
, when we do 11 ^ 11
the result is 0
:
const n = 3;
const result = n ^ n; // 0
const n = 3;
const result = n ^ n; // 0
In other words, an XOR operation of a number with itself will result in 0.
If we do XOR with each number in the array with the indices, eventually all of them will cancel out (result in 0), leaving only the missing number.
You might think that not all numbers are at their index, for example if nums
is [3, 0, 1]
, it is obvious that 3
does not even have an "index 3" that can be associated with it!
For that, we can start by initializing our result to nums.length
. Now, even if the missing number is equal to nums.length
, we are handling that edge case.
let result = nums.length;
let result = nums.length;
Also, XOR is commutative and associative, so it's not important at which index a number appears (or doesn't have one, like in the example above) — they eventually will cancel out.
Now, with a for
loop, we can do the XOR using the bitwise XOR assignment operator:
for (let i = 0; i < nums.length; i++) {
result ^= i ^ nums[i];
}
for (let i = 0; i < nums.length; i++) {
result ^= i ^ nums[i];
}
And, the final result is the missing number. The solution overall looks like this:
function missingNumber(nums: number[]): number {
let result = nums.length;
for (let i = 0; i < nums.length; i++) {
result ^= i ^ nums[i];
}
return result;
}
function missingNumber(nums: number[]): number {
let result = nums.length;
for (let i = 0; i < nums.length; i++) {
result ^= i ^ nums[i];
}
return result;
}
Time and space complexity
The time complexity is again as we iterate through each number in the array, but the space complexity will be as we don't have any additional storage need that will grow as the input size increases.
Next up, we will take a look at the final problem of the entire series, Sum of Two Integers. Until then, happy coding.