2784. Check if Array is Good
Problem Description
The problem presents an integer array nums
and defines a specific type of array called base[n]
. The base[n]
is an array of length n + 1
that contains each integer from 1
to n - 1
exactly once and includes the integer n
exactly twice. For example, base[3]
would be [1, 2, 3, 3]
. The primary task is to determine if the given array nums
is a permutation of any base[n]
array.
A permutation in this context means any rearrangement of the elements. So, if nums
is a rearrangement of all the numbers from 1
to n - 1
and the number n
appears exactly twice, the array is considered "good," and the function should return true
, otherwise false
. The problem simplifies to checking whether the array contains the correct count of each number to match the base
definition.
Intuition
The intuition behind the solution involves counting the occurrences of each number in the array nums
.
-
Identify
n
as the length of the input arraynums
minus one, since abase[n]
array has a length ofn + 1
. This identification is based on the definition mentioned in the problem description. -
Utilize a Counter (a collection type that conveniently counts occurrences of elements) to tally how often each number appears in
nums
. -
Subtract 2 from the count of
n
becausen
should appear exactly twice in a goodbase
array. -
Then subtract 1 from the counts of all other numbers from
1
ton - 1
because each of these numbers should appear exactly once in a goodbase
array. -
Finally, check if all counts are zero using
all(v == 0 for v in cnt.values())
. If they are, it means that the input array has the exact count for each number as required for it to be a permutation of abase[n]
array, and the function should returntrue
. If even one count is not zero, it indicates that there's a discrepancy in the required number frequencies, and the function should returnfalse
.
Learn more about Sorting patterns.
Solution Approach
The implementation of the solution utilizes a standard Python library called collections.Counter
, which is a subclass of dictionary specifically designed to count hashable objects.
Here's the step-by-step breakdown:
-
Compute
n
as the length of the arraynums
minus one. This is because we expect the array to be a permutation of abase[n]
, which has lengthn + 1
.1n = len(nums) - 1
-
Initialize a Counter with
nums
which automatically counts the frequency of each element in the array.1cnt = Counter(nums)
-
Adjust the counted frequencies to match the expectations of a
base[n]
array. According tobase[n]
, the numbern
should appear twice, and all the numbers from1
ton - 1
should appear once. To reflect this in our counter, we subtract 2 from the count ofn
and subtract 1 from the counts of all other numbers within the range1
ton
.1cnt[n] -= 2 2for i in range(1, n): 3 cnt[i] -= 1
-
After the adjustments, a "good" array would leave all counts in the Counter at zero. Verify that this is true by applying the
all
function on a generator expression that checks if allvalues
in the Counter are zero.1return all(v == 0 for v in cnt.values())
-
If any value in the Counter is not zero, then the array cannot be a permutation of "base" because it does not contain the correct frequency of numbers. In such a case, the function will return
false
.
This solution approach utilizes the Counter data structure to perform frequency counting efficiently. The adjustment steps ensure that the counts match the unique base[n]
array's requirements. The final all
check succinctly determines the validity of nums
being a good array, keeping the implementation both effective and elegant.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's use an example to illustrate the solution approach.
Example Input
Consider the array nums = [3, 1, 2, 3]
.
Steps
-
First, calculate
n
by taking the length ofnums
and subtracting one to account for the fact that there should ben + 1
elements in a goodbase
array. In this case,len(nums) - 1
equals4 - 1
which is3
. Hence,n = 3
.1n = len(nums) - 1 # n = 3
-
Initialize a Counter with the array
nums
. This will count how many times each number appears innums
.1cnt = Counter(nums) # cnt = {3: 2, 1: 1, 2: 1}
-
Alter the counted frequencies to mimic a
base[n]
array. The numbern
should appear twice, so we subtract 2 from its count. All other numbers from1
ton - 1
should appear once, so we subtract 1 from their counts.1cnt[n] -= 2 # cnt[3] -= 2 gives cnt = {3: 0, 1: 1, 2: 1} 2for i in range(1, n): 3 cnt[i] -= 1 # iterating and subtracting 1, cnt = {3: 0, 1: 0, 2: 0}
-
All values in the Counter should now be zero for a good
base
array. Using theall
function we check each value:1return all(v == 0 for v in cnt.values()) # This evaluates to `True`
-
Since each number from
1
ton - 1
is included exactly once andn
is included exactly twice, and all adjusted counts are zero, the function will returntrue
. This meansnums
is indeed a permutation of abase[3]
array.
Following this approach, the given array nums = [3, 1, 2, 3]
is confirmed to be a good array and thus the expected output for this example would be True
as it fits the criteria of a base[n]
array permutation.
Solution Implementation
1from collections import Counter # Import the Counter class from the collections module
2
3class Solution:
4 def isGood(self, nums: List[int]) -> bool:
5 # Compute the length of the input array minus one
6 length_minus_one = len(nums) - 1
7
8 # Create a counter to record the frequency of each number in the input array
9 num_counter = Counter(nums)
10
11 # Decrease the count of the number 'length_minus_one' in the counter by 2
12 num_counter[length_minus_one] -= 2
13
14 # Iterate through the range from 1 to 'length_minus_one'
15 for i in range(1, length_minus_one):
16 # Decrease the count of 'i' in the counter by 1
17 num_counter[i] -= 1
18
19 # Return True if all counts in the counter are zero, else return False
20 return all(count == 0 for count in num_counter.values())
21
1class Solution {
2
3 // Method to check if the array 'nums' meets a certain condition
4 public boolean isGood(int[] nums) {
5 // Assuming nums.length - 1 is the maximum number that can be in 'nums'
6 int n = nums.length - 1;
7
8 // Create a counter array with size enough to hold numbers up to 'n'
9 int[] count = new int[201]; // Assumes the maximum value in nums is less than or equal to 200
10
11 // Count the occurrences of each number in nums and store in 'count'
12 for (int number : nums) {
13 ++count[number];
14 }
15
16 // Decrement the count of the last number 'n' by 2 as per the assumed constraint
17 count[n] -= 2;
18
19 // Decrement the count of numbers from 1 to n-1 (inclusive) by 1
20 for (int i = 1; i < n; ++i) {
21 count[i] -= 1;
22 }
23
24 // Check for any non-zero values in 'count', which would indicate 'nums' did not meet the condition
25 for (int c : count) {
26 if (c != 0) {
27 return false;
28 }
29 }
30
31 // If all counts are zero, it means 'nums' meets the condition
32 return true;
33 }
34}
35
1class Solution {
2public:
3 // Function to determine if the given vector 'nums' is "good" by certain criteria.
4 bool isGood(vector<int>& nums) {
5
6 // Calculate the size of 'nums' and store it in 'lastIndex'.
7 int lastIndex = nums.size() - 1;
8
9 // Initialize a counter array 'count' to hold frequencies of numbers in the range [0, 200].
10 vector<int> count(201, 0); // Extended size to 201 to cover numbers from 0 to 200.
11
12 // Populate 'count' vector with the frequency of each number in 'nums'.
13 for (int num : nums) {
14 ++count[num];
15 }
16
17 // The problem description might mention that the last element is counted twice,
18 // so this line compensates for that by decrementing twice.
19 count[lastIndex] -= 2;
20
21 // The problem may specify that we should decrement the frequency
22 // count of all numbers from 1 to 'lastIndex - 1'.
23 for (int i = 1; i < lastIndex; ++i) {
24 --count[i];
25 }
26
27 // Check the 'count' vector. If any element is not zero, return false.
28 // An element not being zero would indicate the 'nums' vector is not "good".
29 for (int counts : count) {
30 if (counts != 0) {
31 return false;
32 }
33 }
34
35 // If all elements in 'count' are zero, the vector 'nums' is "good".
36 return true;
37 }
38};
39
1function isGood(nums: number[]): boolean {
2 // Get the size of the input array.
3 const size = nums.length - 1;
4
5 // Initialize a counter array with all elements set to 0.
6 const counter: number[] = new Array(201).fill(0);
7
8 // Count the occurrence of each number in the input array.
9 for (const num of nums) {
10 counter[num]++;
11 }
12
13 // Decrement the count at the index equal to the size of the input array by 2.
14 counter[size] -= 2;
15
16 // Decrement the count for each index from 1 up to size-1.
17 for (let i = 1; i < size; ++i) {
18 counter[i]--;
19 }
20
21 // Check if all counts are non-negative.
22 return counter.every(count => count >= 0);
23}
24
Time and Space Complexity
Time Complexity
The time complexity of the provided function is determined by a few major steps:
n = len(nums) - 1
: This is a constant time operation,O(1)
.cnt = Counter(nums)
: Building the counter object from thenums
list isO(N)
, whereN
is the number of elements innums
.cnt[n] -= 2
: Another constant time operation,O(1)
.- The loop
for i in range(1, n): cnt[i] -= 1
: This will executeN-1
times (sincen = len(nums) - 1
), and each operation inside the loop is constant time, resulting inO(N)
complexity. - The
all
function combined with the generator expressionall(v == 0 for v in cnt.values())
: Since counting the values in aCounter
object and then iterating through them isO(N)
, this step isO(N)
as well.
Adding up all the parts, the overall time complexity is O(N)
+ O(N)
+ O(N)
which simplifies to O(N)
, because in Big O notation we keep the highest order term and drop the constants.
Space Complexity
The space complexity is also determined by a few factors:
cnt = Counter(nums)
: Storing the count of each number innums
requires additional space which is proportional to the number of unique elements innums
. In the worst case, if all elements are unique, this will beO(N)
.- The
for
loop and theall
function does not use extra space that scales with the size of the input, as they only modify the existingCounter
object.
Therefore, the space complexity is O(N)
, where N
is the number of elements in nums
and assuming all elements are unique.
Learn more about how to find time and space complexity quickly using problem constraints.
Which of the following is a min heap?
Recommended Readings
Sorting Summary Comparisons We presented quite a few sorting algorithms and it is essential to know the advantages and disadvantages of each one The basic algorithms are easy to visualize and easy to learn for beginner programmers because of their simplicity As such they will suffice if you don't know any advanced
Patterns The Shortest Path Algorithm for Coding Interviews The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we can determine the
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
Got a question?ย Ask the Monster Assistantย anything you don't understand.
Still not clear? ย Submitย the part you don't understand to our editors. Or join ourย Discord and ask the community.