经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » 编程经验 » 查看文章
P5369 [PKUSC2018]最大前缀和 - Hs-black
来源:cnblogs  作者:Hs-black  时间:2019/10/8 9:09:57  对本文有异议

状态压缩

P5369

题意:求所有排列下的最大前缀和之和

一步转化: 求最大前缀和的前缀由数集S组成的方案数, 统计答案时直接乘上sum(S)即可

考虑最大前缀和的性质:

设最大前缀和为sum[i]

  1. 到i的后缀均为正数
  2. i后的前缀均为负数

令sum[i] = 集合 i 内所有数的和。

令f[i] = 集合 i内的数组成的排列,最大前缀和 = sum[i]的方案数。

令g[i] = 集合 i内的数组成的排列,所有的最大前缀和都 < 0 的方案数。

代码:

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. using namespace std;
  5. const int N = 25;
  6. const int P = 998244353;
  7. int n, a[N];
  8. int f[1050050], g[1050050];
  9. int sum[1050050];
  10. inline int to(int x) {
  11. return 1 << x;
  12. }
  13. int main() {
  14. cin >> n; int all = to(n) - 1;
  15. for (int i = 1;i <= n; i++)
  16. cin >> a[i], f[to(i-1)] = 1, sum[to(i-1)] = a[i];
  17. for (int i = 1;i <= all; i++)
  18. sum[i] = sum[(i & -i)] + sum[i ^ (i & -i)];
  19. g[0] = 1;
  20. for (int i = 0;i < all; i++) {
  21. if (sum[i] >= 0) {
  22. for (int j = 1;j <= n; j++)
  23. if (!(i & to(j-1)))
  24. f[i | to(j-1)] = ((long long)f[i] + f[i | to(j-1)]) % P;
  25. }
  26. else {
  27. for (int j = 1;j <= n; j++)
  28. if (i & (to(j-1)))
  29. g[i] = ((long long)g[i] + g[i ^ to(j-1)]) % P;
  30. }
  31. }
  32. long long ans = 0;
  33. for (int i = 1;i <= all; i++)
  34. ans = (ans + (long long)f[i] * g[all^i] % P * sum[i] % P) % P;
  35. cout << (ans % P + P) % P << endl;
  36. return 0;
  37. }

原文链接:http://www.cnblogs.com/Hs-black/p/11626107.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号