forked from qiurunze123/leetcode
-
Notifications
You must be signed in to change notification settings - Fork 0
/
lc416.java
70 lines (68 loc) · 2.2 KB
/
lc416.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
package code;
import java.util.HashSet;
/*
* 416. Partition Equal Subset Sum
* 题意:一个数组,可否分成和相等的两部分
* 难度:Medium
* 分类:Dynamic Programming
* 思路:题意可以转换为用任意个元素组成的和等于数组和/2。可以和 lc1, lc15 3-Sum 对比。
* 0,1背包问题,递推比较简单,所以空间可以压缩成一维
* 自己想的思路其实和压缩后的0,1背包类似,但没想到该问题可以抽象为0,1背包
* dp[i][j] = dp[i-1][j] || dp[i-1][j-nums[i]]
* Tips:lc416, lc494
*/
public class lc416 {
public static void main(String[] args) {
int[] nums = {1, 2, 5};
System.out.println(canPartition(nums));
}
public static boolean canPartition(int[] nums) { //自己写的不知道什么,回头看已经看不懂自己的代码了。。。
int sum = 0;
for (int i : nums) {
sum+=i;
}
if(sum%2==1)
return false;
sum/=2;
HashSet<Integer> s = new HashSet();
for (int i = 0; i < nums.length ; i++) {
if(nums[i]==sum)
return true;
HashSet<Integer> s2 = new HashSet(); // 新建一个set,用以存放这一轮的结果
s2.add(nums[i]);
for(int j: s){
if(j+nums[i]==sum)
return true;
s2.add(j+nums[i]);
}
s.addAll(s2);
}
return false;
}
public boolean canPartition2(int[] nums) {
// check edge case
if (nums == null || nums.length == 0) {
return true;
}
// preprocess
int volumn = 0;
for (int num : nums) {
volumn += num;
}
if (volumn % 2 != 0) {
return false;
}
volumn /= 2;
// dp def
boolean[] dp = new boolean[volumn + 1];
// dp init
dp[0] = true;
// dp transition
for (int i = 1; i <= nums.length; i++) {
for (int j = volumn; j >= nums[i-1]; j--) { //从后往前更新,压缩空间
dp[j] = dp[j] || dp[j - nums[i-1]];
}
}
return dp[volumn];
}
}