title | description | hide_table_of_contents | keywords | |||
---|---|---|---|---|---|---|
Combinatorics |
Combinatorics is the branch of mathematics dealing with counting and enumerating the possibilities for a certain event to occur. It is heavily used as it enables us to find very short and concise answers to many problems. |
false |
|
Combinatorics is primarily involved with art of counting. This includes counting number of ways for a certain position to occur, to arrange some objects according to given rules or choosing some objects from a collection. Quite often, the required number takes a very large value, thus it is a very common practice to return the answer by taking modulo with some prime number
Pronounced "n choose r", this is the mathematical notation to represent number of ways to choose r objects out of a collection of n objects. The analytical formula can be written as
This leads to a neat recurrence relation:
We can precompute all the required values using the above formula in
Example #1: 2221 - Find Triangular Sum of an Array
The important insight here is that the figure provided is nothing but an inverted Pascal's Triangle and contribution of each cell in the final sum is the value of cell multiplied by the binomial coefficient at the particular position in Pascal's Triangle.
Thus for the cell at
class Solution {
public:
int triangularSum(vector<int>& nums) {
int n = nums.size();
vector<int> pascalTriangleRow = {1};
// calculate the ith row using (i - 1)th row
for (int i = 0; i < n; i++) {
vector<int> nextRow = {1};
for(int j = 1; j < i; j++){
nextRow.push_back((pascalTriangleRow[j] + pascalTriangleRow[j - 1]) % 10);
}
nextRow.push_back(1);
pascalTriangleRow = nextRow;
}
// calculate the final answer as discussed above
int ans = 0;
for (int i = 0; i < n; i++) {
ans += (nums[i] * pascalTriangleRow[i]) % 10;
}
return ans%10;
}
};
Sometimes it is not possible to calculate the entirety of Pascal's Triangle due to larger values of
The implementation of above can be as follows:
struct comb{
int mod;
// make arrays to store the factorial and inverse factorial modulo m
vector<long long int> factorial;
vector<long long int> inverse_factorial;
// N is the maximum value possible of input
comb (int N, int mod_in = 1e9 + 7) {
// calculate values for factorial
mod = mod_in;
factorial.push_back(1);
for(int i = 1; i <= N; i++) factorial.push_back((factorial.back() * i) % mod);
// calculate values for inverse factorial
vector<long long int> inverse;
inverse.push_back(1);
inverse.push_back(1);
inverse_factorial.push_back(1);
inverse_factorial.push_back(1);
for (int i = 2; i <= N; i++) {
inverse.push_back((mod - ((mod/i) * inverse[mod%i]) % mod) % mod);
inverse_factorial.push_back((inverse_factorial[i - 1] * inverse[i]) % mod);
}
}
// function to calculate nCr(n, r)
long long int nCr(int n, int r){
if ((r < 0) || (r > n)) return 0;
return ((factorial[n] * inverse_factorial[r]) % mod * inverse_factorial[n - r]) % mod;
}
};
For further reading, you can visit cp-algorithms.
This is a very famous sequence of natural numbers and has a variety of applications.
- Number of ways to make balanced bracket sequences using
$n$ left and$n$ right brackets. - Number of ways to make binary trees
- Number of ways to form a mountain range with
$n$ upstrokes and downstrokes. $\$
Here is a more exhastive list.
The
Example #2: 1863 - Sum of All Subset XOR Totals
This is an example of a very tricky problem which heavily simplifies after using some Combinatorics and Bit Manipulation
Here we will consider the
Hence if there are
Thus we can find
Hence all we need to do is find bits which are set atleast once (by computing OR) and then multiply the final answer with
class Solution {
public:
int subsetXORSum(vector<int>& nums) {
int arrayOR = 0;
// do OR of whole array to obtain bits which are set atleast once
for (int num : nums) arrayOR |= num;
// compute the final answer using the formula discussed
return arrayOR * (1 << (nums.size() - 1));
}
};
Example #3: 0062 - Unique Paths
Here our robot always goes either down or right. We know that we have to go down
Notice that we are not required to return the value after taking modulo and the constraints allow for a
class Solution {
public:
int uniquePaths(int m, int n) {
// the upper limit is m + n - 2 = 198
vector<vector<long long int>> PascalTriangle(199, vector<long long int> ());
PascalTriangle[0] = {1};
// calculating every row of the triangle
for (int i = 1; i <= 198; i++) {
PascalTriangle[i].push_back(1);
// using the recurrence relation.
for (int j = 1; j < i; j++) {
// take mod with INT_MAX as otherwise some values may overflow.
PascalTriangle[i].push_back((PascalTriangle[i - 1][j] + PascalTriangle[i - 1][j - 1]) % INT_MAX);
}
PascalTriangle[i].push_back(1);
}
// query the final answer
return PascalTriangle[m + n - 2][n - 1];
}
};
NOTE: Since every testcase only asks us to find
You can check the complete solution for this problem here
Let's represent going left as
Here we can immediately see that such will be impossible in only 2 cases:
- The parity of
$k$ and$endPos - startPos$ is different. - The magnitude of
$k$ is less than magnitude of$endPos - startPos$ .
After checking for above 2 cases, we know for sure that there exists a solution. Now we can just find the number of
Here
Adding both equations,
Thus,
Similarly, by subtracting the equations and simplifying,
Then the solution is
Thus, we need to find
To implement this, you can both precompute the entire Pascal's Triangle, or use concept of mudular inverses to find the required value.
You can check the complete solution for this problem here
export const suggestedProblems = [ { "problemName": "920 - Number of Music Playlists", "difficulty": "Hard", "leetCodeLink": "https://leetcode.com/problems/number-of-music-playlists/", "solutionLink": "" }, { "problemName": "1916 - Count Ways to Build Rooms in an Ant Colony", "difficulty": "Hard", "leetCodeLink": "https://leetcode.com/problems/count-ways-to-build-rooms-in-an-ant-colony/", "solutionLink": "../../solutions/1900-1999/count-ways-to-build-rooms-in-an-ant-colony" }, { "problemName": "1467 - Probability of a Two Boxes Having The Same Number of Distinct Balls", "difficulty": "Hard", "leetCodeLink": "https://leetcode.com/problems/probability-of-a-two-boxes-having-the-same-number-of-distinct-balls/", "solutionLink": "" }, ]