Solution Count of Subset Sum
Problem Statement
Given a set of positive numbers, find the total number of subsets whose sum is equal to a given number 'S'.
Example 1:
Input: {1, 1, 2, 3}, S=4
Output: 3
The given set has '3' subsets whose sum is '4': {1, 1, 2}, {1, 3}, {1, 3}
Note that we have two similar sets {1, 3}, because we have two '1' in our input.
Example 2:
Input: {1, 2, 7, 1, 5}, S=9
Output: 3
The given set has '3' subsets whose sum is '9': {2, 7}, {1, 7, 1}, {1, 2, 1, 5}
Basic Solution
This problem follows the 0/1 Knapsack pattern and is quite similar to "Subset Sum". The only difference in this problem is that we need to count the number of subsets, whereas in the "Subset Sum" we only wanted to know if there exists a subset with the given sum.
A basic brute-force solution could be to try all subsets of the given numbers to count the subsets that have a sum equal to 'S'. So our brute-force algorithm will look like:
for each number 'i' create a new set which includes number 'i' if it does not exceed 'S', and recursively process the remaining numbers and sum create a new set without number 'i', and recursively process the remaining numbers return the count of subsets who has a sum equal to 'S'
Code
Here is the code for the brute-force solution:
class Solution {
public int countSubsets(int[] num, int sum) {
return this.countSubsetsRecursive(num, sum, 0);
}
private int countSubsetsRecursive(int[] num, int sum, int currentIndex) {
// base checks
if (sum == 0) return 1;
if (num.length == 0 || currentIndex >= num.length) return 0;
// recursive call after selecting the number at the currentIndex
// if the number at currentIndex exceeds the sum, we shouldn't process this
int sum1 = 0;
if (num[currentIndex] <= sum) sum1 = countSubsetsRecursive(
num,
sum - num[currentIndex],
currentIndex + 1
);
// recursive call after excluding the number at the currentIndex
int sum2 = countSubsetsRecursive(num, sum, currentIndex + 1);
return sum1 + sum2;
}
public static void main(String[] args) {
Solution ss = new Solution();
int[] num = { 1, 1, 2, 3 };
System.out.println(ss.countSubsets(num, 4));
num = new int[] { 1, 2, 7, 1, 5 };
System.out.println(ss.countSubsets(num, 9));
}
}
The time complexity of the above algorithm is exponential
Top-down Dynamic Programming with Memoization
We can use memoization to overcome the overlapping sub-problems. We will be using a two-dimensional array to store the results of solved sub-problems. As mentioned above, we need to store results for every subset and for every possible sum.
Code
Here is the code:
class Solution {
public int countSubsets(int[] num, int sum) {
Integer[][] dp = new Integer[num.length][sum + 1];
return this.countSubsetsRecursive(dp, num, sum, 0);
}
private int countSubsetsRecursive(
Integer[][] dp,
int[] num,
int sum,
int currentIndex
) {
// base checks
if (sum == 0) return 1;
if (num.length == 0 || currentIndex >= num.length) return 0;
// check if we have not already processed a similar problem
if (dp[currentIndex][sum] == null) {
// recursive call after choosing the number at the currentIndex
// if the number at currentIndex exceeds the sum, we shouldn't process this
int sum1 = 0;
if (num[currentIndex] <= sum) sum1 = countSubsetsRecursive(
dp,
num,
sum - num[currentIndex],
currentIndex + 1
);
// recursive call after excluding the number at the currentIndex
int sum2 = countSubsetsRecursive(dp, num, sum, currentIndex + 1);
dp[currentIndex][sum] = sum1 + sum2;
}
return dp[currentIndex][sum];
}
public static void main(String[] args) {
Solution ss = new Solution();
int[] num = { 1, 1, 2, 3 };
System.out.println(ss.countSubsets(num, 4));
num = new int[] { 1, 2, 7, 1, 5 };
System.out.println(ss.countSubsets(num, 9));
}
}
Bottom-up Dynamic Programming
We will try to find if we can make all possible sums with every subset to populate the array db[TotalNumbers][S+1].
So, at every step we have two options:
- Exclude the number. Count all the subsets without the given number up to the given sum =>
dp[index-1][sum] - Include the number if its value is not more than the 'sum'. In this case, we will count all the subsets to get the remaining sum =>
dp[index-1][sum-num[index]]
To find the total sets, we will add both of the above two values:
dp[index][sum] = dp[index-1][sum] + dp[index-1][sum-num[index]])
Let's start with our base case of size zero:


![sum: 1, index:1=> (dp[index-1][sum] + dp[index-1][sum - 1])](/assets/85f745584e58e2f8.webp)
![sum: 2, index:1=> (dp[index-1][sum] + dp[index-1][sum - 1])](/assets/ed2fe24405d651f6.webp)
![sum: 4, index:3=> (dp[index-1][sum] + dp[index-1][sum - 3])](/assets/c89c9ea1d8747527.webp)
Code
Here is the code for our bottom-up dynamic programming approach:
class Solution {
public int countSubsets(int[] num, int sum) {
int n = num.length;
int[][] dp = new int[n][sum + 1];
// populate the sum=0 columns, as we will always have an empty set for zero sum
for (int i = 0; i < n; i++) dp[i][0] = 1;
// with only one number, we can form a subset only when the required sum is equal to its value
for (int s = 1; s <= sum; s++) {
dp[0][s] = (num[0] == s ? 1 : 0);
}
// process all subsets for all sums
for (int i = 1; i < num.length; i++) {
for (int s = 1; s <= sum; s++) {
// exclude the number
dp[i][s] = dp[i - 1][s];
// include the number, if it does not exceed the sum
if (s >= num[i]) dp[i][s] += dp[i - 1][s - num[i]];
}
}
// the bottom-right corner will have our answer.
return dp[num.length - 1][sum];
}
public static void main(String[] args) {
Solution ss = new Solution();
int[] num = { 1, 1, 2, 3 };
System.out.println(ss.countSubsets(num, 4));
num = new int[] { 1, 2, 7, 1, 5 };
System.out.println(ss.countSubsets(num, 9));
}
}
The above solution has time and space complexity of
Challenge
Can we further improve our bottom-up DP solution? Can you find an algorithm that has
Similar to the space optimized solution for "0/1 Knapsack."
class Solution {
static int countSubsets(int[] num, int sum) {
int n = num.length;
int[] dp = new int[sum + 1];
dp[0] = 1;
// with only one number, we can form a subset only when the required sum is equal to its value
for (int s = 1; s <= sum; s++) {
dp[s] = (num[0] == s ? 1 : 0);
}
// process all subsets for all sums
for (int i = 1; i < num.length; i++) {
for (int s = sum; s >= 0; s--) {
if (s >= num[i]) dp[s] += dp[s - num[i]];
}
}
return dp[sum];
}
public static void main(String[] args) {
Solution ss = new Solution();
int[] num = { 1, 1, 2, 3 };
System.out.println(ss.countSubsets(num, 4));
num = new int[] { 1, 2, 7, 1, 5 };
System.out.println(ss.countSubsets(num, 9));
}
}
🤖 Don't fully get this? Learn it with Claude
Stuck on Solution Count of Subset Sum? Open Claude, copy a block below, and it'll teach you this exact concept — visually and interactively.
Progressively stronger hints — you still solve it.
I'm working on the problem **Solution Count of Subset Sum** (DSA). Give me a HINT LADDER: start with the tiniest nudge, then wait. Only reveal the next, stronger hint when I ask. Do NOT show the full solution unless I type 'show solution'. Keep me doing the thinking. If you're unsure or a claim isn't standard, say so and reason from first principles instead of guessing.
See the technique, not just code.
Explain the optimal approach to **Solution Count of Subset Sum** with a VISUAL walkthrough: trace it on a small concrete example using ASCII art / a step-by-step diagram, narrate what changes each step, then give time & space complexity with a one-line derivation. If you're unsure or a claim isn't standard, say so and reason from first principles instead of guessing.
Catch bugs, edge cases, sub-optimality.
I'll paste my solution to **Solution Count of Subset Sum**. Review it for correctness, missed edge cases, and time/space complexity, then coach me toward the optimal — don't just rewrite it. Ask me to paste my code now. If you're unsure or a claim isn't standard, say so and reason from first principles instead of guessing.
Lock in recognition with look-alikes.
Give me 2 problems that use the SAME underlying pattern as **Solution Count of Subset Sum**. For each, let me attempt first, then review my answer and name the trigger signal that reveals the pattern. If you're unsure or a claim isn't standard, say so and reason from first principles instead of guessing.