2032. Two Out of Three
Problem Description
The problem gives us three integer arrays named nums1
, nums2
, and nums3
. Our task is to find a distinct array that contains all the unique values which are present in at least two of these arrays. This means if a value is present in either array nums1
& nums2
, nums1
& nums3
, or nums2
& nums3
, it should be included in the final answer. Also, the result should not have any duplicates, and the order of values in the resulting array does not matter.
Intuition
To solve this problem efficiently, we can use the properties of set data structures in Python, which store only unique elements. First, we convert each input array (nums1
, nums2
, nums3
) to a set to eliminate any duplicates within the individual arrays. This step helps to simplify our upcoming operations since sets do not allow duplicate values, and membership checks (to see if a value is present in the set) are done in constant time.
The next step is to iterate through a range of possible values that could be present in the arrays. Since the problem does not specify an upper limit on the values, we can assume that the numbers are within a reasonable domain. In this solution, it is assumed that the values are within the inclusive range of 1 to 100.
For each number in this range, we check if it is present in at least two of the three sets. To do this, we find the sum of the boolean results of the membership tests for the three sets (i.e., (i in s1) + (i in s2) + (i in s3)
). If a number is present in at least two sets, this sum will be greater than 1. We use a list comprehension to build the final list of numbers that meet this criterion.
This approach is concise and leverages the strengths of Python's data structures to arrive at a solution that is not only efficient but also easy to understand.
Solution Approach
The implementation of the solution is straightforward yet efficient, leveraging the power of Python's set data structures and list comprehensions. Here's a step-by-step breakdown of the algorithm:
-
Convert Lists to Sets: Since we are only interested in the presence of values rather than their frequency, the first step is to convert the given lists
nums1
,nums2
, andnums3
into sets. This is done using the Python built-in set constructor. By doing so, we remove any duplicates within each of the individual lists, and it also allows us to perform the next steps more efficiently.1s1, s2, s3 = set(nums1), set(nums2), set(nums3)
-
Iterate Over a Range of Values: Given the problem does not provide a specific range for the integer values, the solution assumes a reasonable range from 1 to 100, inclusive. Since there's no information that any value outside this range will appear in the arrays, it is safe to iterate over these 100 numbers.
1range(1, 101)
-
Check for Presence in Two Sets: For each value
i
in the range, we check ifi
is present in at least two out of the three sets (s1
,s2
,s3
). This is done using a simple sum of boolean expressions:1(i in s1) + (i in s2) + (i in s3) > 1
Each expression
(i in s1)
,(i in s2)
, or(i in s3)
returns a Boolean valueโTrue
ifi
is a member of the set andFalse
otherwise. When these are added together,True
counts as1
andFalse
as0
. If the sum is greater than 1, it indicates thati
is present in at least two sets. -
List Comprehension to Create the Result: A list comprehension is used to iterate through the range and apply the check. Only values that pass the check (present in at least two sets) are included in the final list. The beauty of using a list comprehension here is that it returns a new list directly, without having to manually initialize an empty list and append qualifying values one by one.
1[i for i in range(1, 101) if (i in s1) + (i in s2) + (i in s3) > 1]
By incorporating these steps into the provided twoOutOfThree
method of the Solution
class, we get a compact yet elegant solution that meets the problem's requirements effectively. This solution approach demonstrates the strength of using sets for membership tests, the efficiency of list comprehensions for building lists, and some basic mathematical operations (like summing boolean values) for conditional checks.
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 examine the solution approach with a small example. Assume the input arrays are as follows:
1nums1 = [1, 2, 3] 2nums2 = [2, 4] 3nums3 = [3, 4, 5]
Applying the solution approach step-by-step:
-
Convert Lists to Sets: We convert
nums1
,nums2
, andnums3
to sets to remove any duplicates and to simplify the upcoming membership checks:1s1 = set(nums1) # s1 = {1, 2, 3} 2s2 = set(nums2) # s2 = {2, 4} 3s3 = set(nums3) # s3 = {3, 4, 5}
-
Iterate Over a Range of Values: We loop through the range from 1 to 100, as we are assuming a reasonable domain for the numbers:
1range(1, 101) # Will iterate from 1 to 100 inclusive
-
Check for Presence in Two Sets: For each number
i
from 1 to 100, we perform the membership tests to see ifi
is present in at least two of the setss1
,s2
, ands3
. For example:1(2 in s1) + (2 in s2) + (2 in s3) # This equals 2, since 2 is in s1 and s2 2(4 in s1) + (4 in s2) + (4 in s3) # This equals 2, since 4 is in s2 and s3
Only the numbers meeting the condition
> 1
will signify they are present in at least two arrays. -
List Comprehension to Create the Result: We use a list comprehension to go through the range and collect numbers that meet our condition:
1[i for i in range(1, 101) if (i in s1) + (i in s2) + (i in s3) > 1]
Given our example arrays, the final processing would look like this:
1s1, s2, s3 = set(nums1), set(nums2), set(nums3)
2result = [i for i in range(1, 101) if (i in s1) + (i in s2) + (i in s3) > 1]
The numbers that are present in at least two of the sets are 2, 3, and 4. Therefore, the result
would be:
1result = [2, 3, 4]
This small example illustrates how the four steps of the solution approach come together to solve the problem efficiently. Using this method, we quickly find the numbers that are present in at least two of the input arrays and return them as part of the distinct result array.
Solution Implementation
1class Solution:
2 def twoOutOfThree(self, nums1: List[int], nums2: List[int], nums3: List[int]) -> List[int]:
3 # Convert each list into a set to remove duplicates and perform efficient lookups
4 set_nums1, set_nums2, set_nums3 = set(nums1), set(nums2), set(nums3)
5
6 # Create a list comprehension to find numbers present in at least two out of the three sets
7 # The range 1 to 101 is used based on the problem description, which implies that numbers
8 # between 1 and 100 (inclusive) should be considered
9 common_numbers = [
10 number for number in range(1, 101)
11 if (number in set_nums1) + (number in set_nums2) + (number in set_nums3) > 1
12 ]
13
14 # Return the list of numbers found in at least two of the three sets
15 return common_numbers
16
1class Solution {
2 // Main method that finds the common elements present in at least two out of the three arrays
3 public List<Integer> twoOutOfThree(int[] nums1, int[] nums2, int[] nums3) {
4 // Get the frequency array for each input array
5 int[] frequencyArray1 = getFrequencyArray(nums1);
6 int[] frequencyArray2 = getFrequencyArray(nums2);
7 int[] frequencyArray3 = getFrequencyArray(nums3);
8
9 // Initialize the list to store the result
10 List<Integer> result = new ArrayList<>();
11
12 // Traverse the frequency arrays and check if a number is present in at least two arrays
13 for (int i = 1; i <= 100; ++i) {
14 // If the sum of frequencies at this index is greater than 1, it's present in at least two arrays
15 if (frequencyArray1[i] + frequencyArray2[i] + frequencyArray3[i] > 1) {
16 result.add(i); // Add to the result list
17 }
18 }
19
20 return result; // Return the list of numbers present in at least two out of the three arrays
21 }
22
23 // Helper method to create a frequency array for a given input array
24 private int[] getFrequencyArray(int[] nums) {
25 // Array of size 101, since the range is from 1 to 100 inclusive
26 int[] frequency = new int[101];
27
28 // Populate the frequency array; mark 1 for each number that appears in the array
29 for (int num : nums) {
30 frequency[num] = 1; // Mark the occurrence of 'num' by setting the corresponding index to 1
31 }
32
33 return frequency; // Return the frequency array
34 }
35}
36
1#include <vector>
2
3class Solution {
4public:
5 // Rename 'get' to a more descriptive function name 'calculateFrequencyCount'.
6 // This function takes a vector of integers and returns a frequency count vector
7 // where the index corresponds to the integer and the value is 1 if the integer is present.
8 vector<int> calculateFrequencyCount(vector<int>& nums) {
9 vector<int> count(101, 0); // Initialize frequency count with zeros for up to 100 unique integers.
10 for (int num : nums) {
11 count[num] = 1; // Mark the presence of an integer with 1.
12 }
13 return count;
14 }
15
16 // This function takes three integer vectors and returns a vector containing
17 // integers that appear in at least two of the three input vectors.
18 vector<int> twoOutOfThree(vector<int>& nums1, vector<int>& nums2, vector<int>& nums3) {
19 // Calculate frequency count for each input vector.
20 vector<int> countNums1 = calculateFrequencyCount(nums1);
21 vector<int> countNums2 = calculateFrequencyCount(nums2);
22 vector<int> countNums3 = calculateFrequencyCount(nums3);
23
24 vector<int> result; // Initialize an empty vector to store the result.
25
26 // Loop through possible integer values (1 to 100).
27 for (int i = 1; i <= 100; ++i) {
28 // If an integer appears in at least two out of the three vectors,
29 // add it to the result vector.
30 if (countNums1[i] + countNums2[i] + countNums3[i] > 1) {
31 result.emplace_back(i);
32 }
33 }
34
35 return result; // Return the result vector.
36 }
37};
38
1function twoOutOfThree(nums1: number[], nums2: number[], nums3: number[]): number[] {
2 // Initialize an array 'counts' to store the frequency of each number (0 - 100)
3 // assuming the values in the input arrays are within 1 - 100 based on the problem constraints.
4 const counts = new Array(101).fill(0);
5
6 // Loop through unique values of nums1 and increment the count
7 new Set(nums1).forEach(val => counts[val]++);
8
9 // Loop through unique values of nums2 and increment the count
10 new Set(nums2).forEach(val => counts[val]++);
11
12 // Loop through unique values of nums3 and increment the count
13 new Set(nums3).forEach(val => counts[val]++);
14
15 // Initialize an array 'result' to store the numbers that appear in at least two arrays
16 const result = [];
17
18 // Iterate over 'counts' and push the numbers with a count of 2 or more to 'result'
19 counts.forEach((val, idx) => {
20 if (val >= 2) {
21 result.push(idx);
22 }
23 });
24
25 // Return the 'result' array containing the numbers that meet the criteria
26 return result;
27}
28
Time and Space Complexity
The given code aims to find all the numbers present in at least two out of the three input lists, and the solution leverages sets for efficient lookups. Here's the analysis of the time and space complexities:
Time Complexity
- Converting
nums1
,nums2
, andnums3
into setss1
,s2
, ands3
. This step has a time complexity ofO(n)
for each list, wheren
is the length of the longest list amongnums1
,nums2
, andnums3
. - Iterating through the range
[1, 101]
and checking the presence of each number in the sets takesO(100)
time, since it iterates over a fixed range of numbers. - The presence checks
i in s1
,i in s2
, andi in s3
areO(1)
operations due to set lookup properties. Since these checks are done for each number in[1, 101]
, this does not change the overall fixed time iteration.
Considering these steps, the overall time complexity can be approximated to O(n + 100)
, which simplifies to O(n)
because, typically, the constant factors are dropped in complexity analysis.
Space Complexity
- The additional sets
s1
,s2
, ands3
each have a maximum space complexity ofO(n)
, assuming all elements are unique in the original lists. - The final list comprehension doesn't store more than
100
integers since it's constrained by the range[1, 101]
, which accounts forO(1)
space.
Hence, the total space complexity of the code is O(3n + 1)
, which simplifies to O(n)
as constant factors and coefficients are omitted in Big O notation.
Learn more about how to find time and space complexity quickly using problem constraints.
Which algorithm should you use to find a node that is close to the root of the tree?
Recommended Readings
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
Runtime Overview When learning about algorithms and data structures you'll frequently encounter the term time complexity This concept is fundamental in computer science and offers insights into how long an algorithm takes to complete given a certain input size What is Time Complexity Time complexity represents the amount of time
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.