2151. Maximum Good People Based on Statements
Problem Description
In this LeetCode problem, we're given a set of statements made by a group of n people, about each other's trustworthiness. Specifically, the input is a 2D integer array statements
where:
statements[i][j]
= 0 means personi
claims personj
is bad (untruthful).statements[i][j]
= 1 means personi
claims personj
is good (truthful).statements[i][j]
= 2 means personi
has not made any claim about personj
.
Individuals are not permitted to make statements about themselves, thus statements[i][i]
is always 2. The objective is to determine the maximum number of people that can be considered 'good' based on these statements. A 'good' person always tells the truth, while a 'bad' person may lie or tell the truth.
Flowchart Walkthrough
Let's analyze the problem LeetCode 2151. Maximum Good People Based on Statements using the Flowchart. Here's how we can deduce the best approach:
-
Is it a graph?
- No: The problem description doesn't involve a typical graph structure with nodes directly connected by edges.
-
Need to solve for kth smallest/largest?
- No: The task is not about finding a kth smallest or largest element.
-
Involves Linked Lists?
- No: There is no manipulation or traversal of linked lists mentioned.
-
Does the problem have small constraints?
- Yes: Each person can either be good or bad, which suggests a limited number of combinations or permutations to explore, indicative of small constraints that suit exhaustive techniques.
-
Brute force / Backtracking?
- Yes: Given the small constraints and the nature of trying all combinations of good and bad people to validate against given statements, brute force or backtracking is most suited. We need to explore every possible combination of people's states (good or bad) and verify their consistency with provided statements, which is characteristic of backtracking.
Conclusion: By following the path detailed in the flowchart and the problem's requirements, it's clear that using a backtracking pattern to explore all possible states and validate them against provided statements is the most appropriate approach for solving LeetCode 2151. Maximum Good People Based on Statements.
Intuition
The intuition behind the solution is to utilize a brute-force approach, trying every possible combination of 'good' and 'bad' assignments for the people. Since a 'good' person always tells the truth, their statements can be used to cross-verify which people can be good and which cannot. If a 'good' person says another person is good, then both must be good; if they say someone is bad, then that person cannot be considered good in any combination that includes the first person as a good individual.
The solution uses bit masking to represent each combination of 'good' and 'bad' people. Each bit in a mask represents a person, where 1
can denote 'good' and 0
can denote 'bad'. There are 2^n
possible combinations (masks) to check, where n
is the number of people.
The function check(mask)
takes a mask as an argument and iterates over all people. It checks the statements of those who are 'good' (bits set to 1
) in the current mask. For each 'good' person's statements, it checks whether they align with the current assumption of who's good and bad. For instance, if a 'good' person claims another person is good, but the mask marks them as bad (or vice versa), the current mask is invalid, and the function returns 0. Otherwise, if no conflicts are found, it returns the number of 'good' people in the current mask (the count of 1's
in the mask).
The main function calls check(mask)
for all 2^n
combinations and returns the maximum number of good people found among all valid masks that don't lead to contradictions.
This brute-force solution works well for smaller n
, as it explores all possibilities and ensures that the maximum number of 'good' people is found according to the statements made.
Learn more about Backtracking patterns.
Solution Approach
The implementation is grounded on a bit manipulation strategy. It uses an integer to represent a set of truth assignments to individuals, where the i-th
bit of the integer corresponds to the i-th
person.
Here is a step-by-step walkthrough of the solution:
-
Brute Force Enumeration: We generate all possible combinations of each person being 'good' or 'bad' by iterating through integers from
1
to2^n - 1
, inclusive. Here,n
is the number of people. The reason we start from1
is because0
would mean that nobody is good, which is not a valid scenario for us to consider. -
Checking Each Combination: For each combination (represented by
mask
), we implement a function calledcheck(mask)
. This function is responsible for validating the combination against the statements made. -
Validating the 'Good' People: Within
check(mask)
, we traverse over each person (i
) by examining the current bit in the mask:(mask >> i) & 1
. If this results in1
, the person is assumed 'good' in this particular combination. Then, we check each statementi
has made about other people (j
). -
Statement Verification: If person
i
has made a definitive statement about personj
(i.e.,statements[i][j]
is 0 or 1), we check the truthfulness ofi
's statement against the current combination (mask
). If a 'good'i
made a statement that contradicts the current assumption aboutj
, the function returns0
, indicating this mask (or combination) is not valid. -
Counting 'Good' People: If no contradictions are found for a 'good' person
i
, the count increments, tallying the number of 'good' people in this combination. -
Finding the Maximum: Finally, the main solution function calls
check(mask)
for each possibility and keeps track of the maximum number of 'good' people found from all the valid combinations. It uses the built-in Python functionmax()
to find the maximum over all return values fromcheck(mask)
.
By utilizing bit masks to represent sets and checking all possible combinations, the algorithm finds the optimal solution that matches the problem's constraints. This approach, while not the most time-efficient for large n
, ensures accuracy by not overlooking any potential solution when n
is within a manageable size.
Example Walkthrough
Suppose we have a group of 3 people, and we are given the following statements
array:
statements = [ [2, 1, 0], // Person 0's statements about Person 1 and 2 [2, 2, 1], // Person 1's statements about Person 0 and 2 [1, 0, 2] // Person 2's statements about Person 0 and 1 ]
Here's how we would apply the solution approach:
-
Brute Force Enumeration: Since there are 3 people, we have
2^3 = 8
combinations to consider. We will ignore the mask000
(where nobody is good) and check001
to111
. -
Checking Each Combination: Let's consider the mask
011
, indicating Person 0 is 'bad' while Persons 1 and 2 are 'good'. We use thecheck(mask)
function to determine if this mask is a valid representation of the 'good' people. -
Validating the 'Good' People: In the
check(mask)
function, we start with Person 0 being 'bad', so we skip and move to Persons 1 and 2. -
Statement Verification: For Person 1 (who is 'good' in this mask), their statement about Person 2 is that they are 'good' (
statements[1][2]
is 1), which aligns with our mask. However, we do not have a statement from Person 1 about Person 0, so we cannot verify Person 1's truthfulness this way.Moving to Person 2, they state that Person 0 is 'good' (
statements[2][0]
is 1), but our mask assumes Person 0 is 'bad', creating a contradiction. Person 2 also claims that Person 1 is 'bad' (statements[2][1]
is 0), which contradicts our mask where Person 1 is 'good'. -
Counting 'Good' People: Since there's a contradiction in Person 2's statements, we cannot count any 'good' people under this mask, so
check(mask)
returns 0. -
Finding the Maximum: We perform the same
check(mask)
for all other combinations. Let's take another example with the mask101
, where Persons 0 and 2 are 'good', and Person 1 is 'bad'. Person 0 claims Person 1 is 'good' which is a contradiction since we assumed them to be 'bad'. Therefore,check(mask)
also returns 0 for this mask.
By going through all combinations, we may find that the mask 110
yields the maximum number of 'good' people without contradiction — Person 0 claims Person 2 is 'bad', and Person 2 claims Person 0 is 'good', which both align with this mask. In this scenario, both Persons 0 and 1 can be considered good based on their statements and each other's statements.
Therefore, the final result given by the main function would be 2
, as this is the maximum number of 'good' people found across all valid combinations. The function would iterate through all the masks, evaluate each one using the check(mask)
function, and conclude that the combination 110
presents the maximum number of good people without causing any conflicts.
Solution Implementation
1class Solution:
2 def maximumGood(self, statements):
3 # Helper function to check and count the number of 'good' people based on the given mask
4 def check(mask):
5 count_good = 0 # Initialize count of 'good' people
6 # Iterate through each person's statements
7 for idx, statement in enumerate(statements):
8 # Check if the bit in the mask for this person is set to 1 (indicating 'good')
9 if (mask >> idx) & 1:
10 # Iterate through this person's statements about others
11 for jdx, value in enumerate(statement):
12 # If the statement is not 'unknown' and conflicts with the mask, return 0
13 if value < 2 and ((mask >> jdx) & 1) != value:
14 return 0
15 # If there is no conflict, increment the count of 'good' people
16 count_good += 1
17 return count_good # Return the count of 'good' people for this mask
18
19 # Iterate over all possible combinations of 'good' and 'bad' people and find the max count of 'good' people
20 # 1 << len(statements) gives the total number of combinations for 'good' / 'bad' assignments
21 max_good_people = max(check(mask) for mask in range(1, 1 << len(statements)))
22 return max_good_people
23
24# The List type need to be imported from typing if you want to use it in type hints
25from typing import List
26
27# Then you could use it as follows:
28class Solution:
29 def maximumGood(self, statements: List[List[int]]) -> int:
30 # ... rest of the code remains the same
31
1class Solution {
2 // Method to find the maximum number of people that can be classified as good
3 public int maximumGood(int[][] statements) {
4 int maxGoodPersons = 0; // Initialize the count of maximum good persons to zero.
5
6 // Iterate through all possible subsets of persons using a mask
7 for (int mask = 1; mask < (1 << statements.length); ++mask) {
8 // Check the current subset and update the maximum if applicable
9 maxGoodPersons = Math.max(maxGoodPersons, checkSubset(mask, statements));
10 }
11
12 return maxGoodPersons; // Return the maximum number of good persons found
13 }
14
15 // Helper method to check if a given subset of persons is valid and calculate the count
16 private int checkSubset(int mask, int[][] statements) {
17 int goodPersonCount = 0; // Initialize the count of good persons in this subset to zero.
18 int personCount = statements.length; // The total number of persons
19
20 // Iterate over each person to check if they can be classified as good, based on the current subset
21 for (int i = 0; i < personCount; ++i) {
22 // Check if the current person (i) is included in the subset (good) as per the mask
23 if (((mask >> i) & 1) == 1) {
24
25 // Iterate over all the statements made by this person
26 for (int j = 0; j < personCount; ++j) {
27 int statementValue = statements[i][j]; // Statement about person j made by person i
28
29 // If the statement value is less than 2 (0 or 1), check consistency with subset
30 // If there is an inconsistency, the subset is not valid and we return 0
31 if (statementValue < 2 && ((mask >> j) & 1) != statementValue) {
32 return 0;
33 }
34 }
35
36 // If all statements by person i are consistent, increment the count of good persons
37 ++goodPersonCount;
38 }
39 }
40
41 // Return the total number of good persons in this subset if it is valid
42 return goodPersonCount;
43 }
44}
45
1class Solution {
2public:
3 // Finds the maximum number of people who can be good based on their statements
4 int maximumGood(vector<vector<int>>& statements) {
5 int maxGood = 0; // Stores the maximum number of 'good' people found
6
7 // Iterate over all possible combinations of 'good' and 'bad' people
8 for (int mask = 1; mask < (1 << statements.size()); ++mask) {
9 // Compare current count with the new count returned by check function
10 maxGood = max(maxGood, check(mask, statements));
11 }
12
13 return maxGood; // Return the maximum count of 'good' people possible
14 }
15
16 // Checks if a given combination (mask) of 'good' people is consistent with their statements
17 int check(int mask, vector<vector<int>>& statements) {
18 int countGood = 0; // Counter for the number of 'good' people in the current combination
19 int n = statements.size(); // Number of people
20
21 // Iterate over each person
22 for (int i = 0; i < n; ++i) {
23 if ((mask >> i) & 1) { // Check if the person is 'good' in the current mask
24 // If the person is 'good', check all of their statements
25 for (int j = 0; j < n; ++j) {
26 int statementValue = statements[i][j]; // Statement about person j by person i
27
28 // If statementValue < 2, it means person i has made a definitive statement.
29 // Check if the statement is consistent with our current 'good'/'bad' combination
30 if (statementValue < 2 && ((mask >> j) & 1) != statementValue) {
31 return 0; // Inconsistency found; this is not a valid combination
32 }
33 }
34 countGood++; // Increment the count of 'good' people for this combination
35 }
36 }
37
38 return countGood; // Return the number of 'good' people for this combination if consistent
39 }
40};
41
1// Helper function to count the number of 'good' persons that can be inferred based on the current mask.
2// The mask represents a set of people we assume to be 'good' (where each bit corresponds to a person).
3function checkValidity(mask: number, statements: number[][], peopleCount: number): number {
4 let goodCount = 0; // Track number of 'good' people.
5 for (let i = 0; i < peopleCount; ++i) {
6 // If the i-th bit of mask is set, consider the i-th person 'good'.
7 if ((mask >> i) & 1) {
8 for (let j = 0; j < peopleCount; ++j) {
9 const statementValue = statements[i][j];
10 // If a 'good' person makes a definite statement (true or false) about another person,
11 // check if that statement conflicts with our current assumptions. If it does, return 0 as
12 // this configuration of 'good' people is not possible.
13 if (statementValue < 2 && ((mask >> j) & 1) != statementValue) {
14 return 0;
15 }
16 }
17 // If we reach here, person 'i' is 'good' without contradictions.
18 goodCount++;
19 }
20 }
21 // Return the total count of 'good' people in this configuration.
22 return goodCount;
23}
24
25// Main function to find the maximum number of 'good' people based on their statements.
26function maximumGood(statements: number[][]): number {
27 const peopleCount = statements.length; // Total number of people.
28 let maxGood = 0; // Keep track of the maximum number of 'good' people found so far.
29
30 // Iterate over all possible sets of 'good' people represented by masks.
31 // (1 << peopleCount) gives us the number of possible masks, as each person can be good or not.
32 for (let mask = 1; mask < (1 << peopleCount); ++mask) {
33 // Update the maximum good count if the current mask leads to a higher number of 'good' people.
34 maxGood = Math.max(maxGood, checkValidity(mask, statements, peopleCount));
35 }
36 // Return the maximum number of 'good' people that can be deduced without contradictions.
37 return maxGood;
38}
39
Time and Space Complexity
The provided Python code defines a function maximumGood
which finds out the maximum number of people who can be considered "good" based on their statements about themselves and others. The solution employs a brute force approach to check every possible combination of "good" and "bad" people and then validate these combinations against the given statements.
Time Complexity
The time complexity of the code is determined by the number of possible combinations of people being good or bad, since we are iterating over all possible combinations using a bit mask. Given n
people, there are 2^n
possible combinations (since for each person, there is a binary decision to be either good or bad). For each combination, we are checking all statements made by all n
persons, and each person makes n
statements or observations. This results in a nested loop running n
times for each of the 2^n
combinations.
Thus, the time complexity is O(n * 2^n)
, where n
is the number of people.
Space Complexity
The space complexity of the code is quite straightforward. The only additional space used is for the variable cnt
, which is an integer used to keep count of the number of "good" people in the current combination, and the space used in the recursive process stack. This does not scale with n
and hence is O(1)
.
However, it's important to note that the code uses recursive calls to the check
function. The maximum depth of this recursion would be n
in the worst case (although in the check
function there are no recursive calls, it's important to consider for more general cases). Therefore, if we would consider the function call stack in the analysis, the space complexity would be O(n)
for the recursion call stack space. However, since in the provided code check
is more of an iterative function, we consider the space complexity to be O(1)
.
Learn more about how to find time and space complexity quickly using problem constraints.
What's the output of running the following function using the following tree as input?
1def serialize(root):
2 res = []
3 def dfs(root):
4 if not root:
5 res.append('x')
6 return
7 res.append(root.val)
8 dfs(root.left)
9 dfs(root.right)
10 dfs(root)
11 return ' '.join(res)
12
1import java.util.StringJoiner;
2
3public static String serialize(Node root) {
4 StringJoiner res = new StringJoiner(" ");
5 serializeDFS(root, res);
6 return res.toString();
7}
8
9private static void serializeDFS(Node root, StringJoiner result) {
10 if (root == null) {
11 result.add("x");
12 return;
13 }
14 result.add(Integer.toString(root.val));
15 serializeDFS(root.left, result);
16 serializeDFS(root.right, result);
17}
18
1function serialize(root) {
2 let res = [];
3 serialize_dfs(root, res);
4 return res.join(" ");
5}
6
7function serialize_dfs(root, res) {
8 if (!root) {
9 res.push("x");
10 return;
11 }
12 res.push(root.val);
13 serialize_dfs(root.left, res);
14 serialize_dfs(root.right, res);
15}
16
Backtracking Template Prereq DFS with States problems dfs_with_states Combinatorial search problems Combinatorial search problems involve finding groupings and assignments of objects that satisfy certain conditions Finding all permutations combinations subsets and solving Sudoku are classic combinatorial problems The time complexity of combinatorial problems often grows rapidly with the size of
Coding Interview 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
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
Want a Structured Path to Master System Design Too? Don’t Miss This!