2275. Largest Combination With Bitwise AND Greater Than Zero
Problem Description
The challenge is to find the largest combination of numbers from an array candidates
such that when the bitwise AND operation is applied to all numbers in the combination, the result is greater than 0. The bitwise AND of an array is calculated by performing the bitwise AND operation on every integer in the array. Remember, each number in candidates
can be used only once in each combination.
For example, if we have the array [1, 5, 3]
, the bitwise AND is calculated as 1 & 5 & 3
which equals 1
. In the case of a single-element array like [7]
, the bitwise AND is simply that element itself, which is 7
.
The goal is to figure out the size of the largest such combination with a non-zero bitwise AND result.
Intuition
The solution relies on understanding how bitwise AND operations work. When you perform a bitwise AND across multiple numbers, the only way for a bit to remain set (remain 1) in the result is if that bit is set in all the numbers being compared.
Therefore, rather than looking at specific combinations which would lead to a potentially massive computational problem, we need to look at each bit position across all numbers in candidates
.
Here's the intuition behind the solution:
- We loop over each bit position from 0 to 31 (covering all the bits for a standard 32-bit integer) since we are dealing with positive integers.
- In each iteration, we calculate how many numbers in
candidates
have their bit set at the current position. - We keep track of the maximum count of set bits encountered at any position. This maximum count represents the largest combination size, as it shows us the largest group of numbers with a particular bit set. It ensures the bitwise AND will be greater than 0, as at least one bit will be set in the result by all numbers in that combination.
- Finally, we return the largest count which is the size of the largest combination with a bitwise AND greater than 0.
This approach works effectively because it exploits the property of bitwise AND and allows us to avoid explicitly enumerating every combination, which would be impractical for large arrays.
Solution Approach
The implementation of the solution is straightforward, reflecting the intuition discussed earlier. Let's walk through the essential parts of the code and the rationale behind them:
-
Initialization of the answer variable:
ans = 0
- This variable
ans
will hold the maximum count of numbers with a particular bit set that has been encountered so far.
- This variable
-
Iterating through each bit position:
for i in range(32):
- The
for
loop iterates from0
to31
, corresponding to each bit position in a 32-bit integer.
- The
-
Counting set bits at the current position in all candidates:
t = 0 for x in candidates: t += (x >> i) & 1
- We introduce a temporary counter
t
for each bit positioni
. - For each candidate number
x
, we right-shift (>>
) its bits byi
positions, which moves the bit at positioni
to the least significant bit position (rightmost position). - We then perform a bitwise AND with
1
(& 1
), which isolates the least significant bit. - If this bit is set, it contributes
1
to the countt
; otherwise, it contributes0
. - After iterating over all numbers,
t
will hold the total count of numbers with thei
-th bit set.
- We introduce a temporary counter
-
Updating the answer with the maximum count found:
ans = max(ans, t)
- After counting the set bits for the current
i
-th position, we compare this count (t
) with our current maximumans
. - If
t
is greater, it means we've found a larger combination of numbers with theiri
-th bit set, so we updateans
accordingly.
- After counting the set bits for the current
-
Return the largest combination size:
return ans
- Once we have finished checking all bit positions,
ans
holds the size of the largest combination ofcandidates
that can produce a non-zero bitwise AND. - This is the final result, which we return.
- Once we have finished checking all bit positions,
The solution elegantly bypasses the need for combination generation using a frequency count method that capitalizes on the nature of the bitwise AND operation. No additional data structures beyond simple variables are needed, and the solution runs in O(32 * N)
time, where N
is the number of elements in candidates
, which is optimal given the problem constraints.
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 illustrate the solution approach using a small example with the candidates
array [3, 1, 2]
. We want to find the size of the largest combination where the bitwise AND is greater than 0.
-
Initialization of the answer variable: Our starting answer (max count)
ans
is set to0
. -
Iterating through each bit position: We loop over bit positions from
0
to31
. Let's consider the first three positions as an example in this walkthrough. -
Counting set bits at the current position in all candidates:
-
For bit position
0
:3 >> 0
is3
(binary11
),(3 >> 0) & 1
is1
.1 >> 0
is1
(binary01
),(1 >> 0) & 1
is1
.2 >> 0
is2
(binary10
),(2 >> 0) & 1
is0
.- Count
t
becomes2
, as there are two numbers with the least significant bit set.
-
Update answer:
ans = max(0, 2)
which is2
. -
For bit position
1
:3 >> 1
is1
(binary01
),(3 >> 1) & 1
is1
.1 >> 1
is0
(binary00
),(1 >> 1) & 1
is0
.2 >> 1
is1
(binary01
),(2 >> 1) & 1
is1
.- Count
t
is again2
, as there are two numbers with the second least significant bit set.
-
Update answer:
ans = max(2, 2)
does not change and remains2
. -
For bit position
2
:3 >> 2
is0
(binary00
),(3 >> 2) & 1
is0
.1 >> 2
is0
(binary00
),(1 >> 2) & 1
is0
.2 >> 2
is0
(binary00
),(2 >> 2) & 1
is0
.- Count
t
is0
, as none of the numbers have the third bit set.
-
Update answer:
ans = max(2, 0)
remains2
.
-
We would continue this process for all bit positions, but in this case, it's clear that the largest combination size doesn't change and ans
would stay as 2
because there aren't higher bit positions set in any of the numbers in candidates
.
- Return the largest combination size: After checking all possible bit positions in the loop, we find that the answer
ans
is2
, which is the size of the largest combination that can produce a non-zero bitwise AND.
In this example, the combinations [3, 1]
and [1, 2]
have at least one common bit set (either the least or second-least significant bit), and thus both combinations would result in a bitwise AND greater than 0
. No combination of three numbers exists that would satisfy the condition, so 2
is the largest size for such a combination.
Solution Implementation
1class Solution:
2 def largestCombination(self, candidates: List[int]) -> int:
3 # Initialize the maximum count of candidates that have
4 # the same bit set (starting with the count set to 0)
5 max_count = 0
6
7 # Iterate over each bit position (0 to 31 since we're working with 32-bit integers)
8 for bit_position in range(32):
9 # Counter to hold the number of candidates with the current bit set
10 count_with_bit_set = 0
11
12 # Iterate over the candidate numbers
13 for candidate in candidates:
14 # Check if the bit at the current bit position is set (1)
15 if (candidate >> bit_position) & 1:
16 # If so, increment our counter
17 count_with_bit_set += 1
18
19 # Update the maximum count if the current count is greater than the previously recorded maximum
20 max_count = max(max_count, count_with_bit_set)
21
22 # Return the maximum count of candidates that have a particular bit set
23 return max_count
24
1class Solution {
2 // Method to find the largest combination of numbers that have the highest number of
3 // set bits in the same position when aligned in binary representation.
4 public int largestCombination(int[] candidates) {
5 int maxCombinationSize = 0; // This will hold the maximum combination size
6
7 // Iterate through each bit position from 0 to 31 (for 32-bit integers)
8 for (int i = 0; i < 32; ++i) {
9 int count = 0; // Count of candidates that have the current bit set
10
11 // Iterate through each candidate number
12 for (int candidate : candidates) {
13 // Right shift the candidate by i bits and check if the least significant bit is set
14 // If it's set, increment the count for this bit position
15 count += (candidate >> i) & 1;
16 }
17
18 // Compare the count for the current bit position with the maximum combination size
19 // and update the maxCombinationSize if necessary
20 maxCombinationSize = Math.max(maxCombinationSize, count);
21 }
22
23 // Return the maximum combination size found
24 return maxCombinationSize;
25 }
26}
27
1#include <vector>
2#include <algorithm>
3
4class Solution {
5public:
6 int largestCombination(std::vector<int>& candidates) {
7 // Declare 'BITS' to specify the number of bits to check in each number.
8 const int BITS = 24;
9
10 // Initialize 'maxCount' to 0 to keep track of the maximum count of numbers that have the same bit set.
11 int maxCount = 0;
12
13 // Iterate over each bit position from 0 to 'BITS - 1'.
14 for (int i = 0; i < BITS; ++i) {
15 // Initialize 'bitCount' to count the number of candidates with the current bit set.
16 int bitCount = 0;
17
18 // Iterate through each number in the 'candidates' vector.
19 for (int num : candidates) {
20 // Use bitwise operations to check if the bit at position 'i' is set.
21 if ((num >> i) & 1) {
22 // If the current bit is set, increment 'bitCount'.
23 bitCount++;
24 }
25 }
26
27 // Update 'maxCount' if 'bitCount' of the current bit is greater than the 'maxCount' found so far.
28 maxCount = std::max(maxCount, bitCount);
29 }
30
31 // Return the maximum count of numbers that have the same bit set.
32 return maxCount;
33 }
34};
35
1function largestCombination(candidates: number[]): number {
2 // Declare 'BITS' to specify the number of bits to check in each number.
3 const BITS = 24;
4
5 // Initialize 'maxCount' to 0 to keep track of the maximum count of numbers that have the same bit set.
6 let maxCount = 0;
7
8 // Iterate over each bit position from 0 to 'BITS - 1'.
9 for (let i = 0; i < BITS; i++) {
10 // Initialize 'bitCount' to count the number of candidates with the current bit set.
11 let bitCount = 0;
12
13 // Iterate through each number in the candidates array.
14 for (let num of candidates) {
15 // Use bitwise operations to check if the bit at position 'i' is set.
16 if ((num >> i) & 1) {
17 // If the current bit is set, increment 'bitCount'.
18 bitCount++;
19 }
20 }
21
22 // Update 'maxCount' if 'bitCount' of the current bit is greater than the 'maxCount' found so far.
23 maxCount = Math.max(maxCount, bitCount);
24 }
25
26 // Return the maximum count of numbers that have the same bit set.
27 return maxCount;
28}
29
Time and Space Complexity
The time complexity of the given code is O(n * m)
, where n
is the number of elements in candidates
and m
is the number of bits considered for each number (here, m
is fixed at 32, since we're dealing with 32-bit integers). This is because we have one outer loop running for each of the 32 bits, and an inner loop that goes through each of the n
candidates to count the bits at the i-th
position.
The space complexity is O(1)
since we only use a constant amount of extra space, including variables for the answer ans
, the counter t
, and the loop index i
, irrespective of the input size.
Learn more about how to find time and space complexity quickly using problem constraints.
Which technique can we use to find the middle of a linked list?
Recommended Readings
LeetCode Patterns Your Personal Dijkstra's Algorithm to Landing Your Dream Job 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
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
Want a Structured Path to Master System Design Too? Don’t Miss This!