Vijos P1250 - 最勇敢的机器人

思路

用并查集,然后分组背包。

背包九讲造福人类。

分组背包和一般的背包差不多,就是把一组作为一个点考虑。

在如何把它们集中到一组想了很久,后来还是暴力了。先把它们都分好,然后扫描一遍根节点,把相同根节点的都存到一个数组里,然后遍历一遍数组。

代码

 
 
  1. #include <cstdio>
  2. #include <cstring>
  3. #include <algorithm>
  4. #include <map>
  5. #include <queue>
  6. #include <cctype>
  7. #define MP(a, b) make_pair(a, b)
  8. using namespace std;
  9. const int MAXN = 1000 + 5;
  10. const int INF = 0x3f3f3f3f;
  11. struct EQP
  12. {
  13. int val, w;
  14. }eqp[MAXN];
  15. vector<int> ve[MAXN];
  16. int dp[MAXN][10000], pa[MAXN];
  17. int Find(int x)
  18. {
  19. return pa[x] == x ? x : pa[x] = Find(pa[x]);
  20. }
  21. int main()
  22. {
  23. //freopen("input.txt", "r", stdin);
  24. int n, wMax, k, i, j;
  25. scanf("%d%d%d", &n, &wMax, &k);
  26. for (i = 1; i <= n; i++)
  27. scanf("%d%d", &eqp[i].val, &eqp[i].w);
  28. for (i = 1; i <= n; i++)
  29. pa[i] = i;
  30. for (i = 0; i < k; i++)
  31. {
  32. int a, b;
  33. scanf("%d%d", &a, &b);
  34. int x = Find(a), y = Find(b);
  35. if (x != y)
  36. pa[x] = y;
  37. }
  38. for (i = 1; i <= n; i++)
  39. {
  40. int x = Find(i);
  41. ve[x].push_back(i);
  42. }
  43. int cnt = 0;
  44. for (i = 1; i <= n; i++)
  45. {
  46. if (!ve[i].empty())
  47. {
  48. cnt++;
  49. for (j = 0; j <= wMax; j++)
  50. {
  51. dp[cnt][j] = dp[cnt - 1][j];
  52. for (int k = 0; k < ve[i].size(); k++)
  53. if (j >= eqp[ve[i][k]].w)
  54. dp[cnt][j] = max(dp[cnt][j], dp[cnt - 1][j - eqp[ve[i][k]].w] + eqp[ve[i][k]].val);
  55. }
  56. }
  57. }
  58. printf("%d\n", dp[cnt][wMax]);
  59. return 0;
  60. }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值