题目传送门:LOJ #3158

题意简述:

给定两个长度为 \(n\) 的正整数序列 \(a,b\),要求在每个序列中都选中 \(K\) 个下标,并且要保证同时在两个序列中都被选中的下标至少有 \(L\) 个,使得选中的下标对应的数的总和最大。

题解:

题目相当于要求在两个序列中选出 \(K\) 对数,不妨一对一对地选。

有个结论是说,上一步的最优决策一定不会再反悔,就是已经选的不会再撤销。

然后做完了,用堆维护一些东西,精细实现就好了。

下面是代码,复杂度 \(\mathcal{O}\left(\sum n\log n\right)\)。

  1. #include <cstdio>
  2. #include <algorithm>
  3. #include <queue>
  4. #include <cctype>
  5. #define fi first
  6. #define se second
  7. #define mp std::make_pair
  8. typedef long long LL;
  9. typedef std::pair<int, int> pii;
  10. typedef std::priority_queue<pii> PQ;
  11. const int MN = 200005;
  12. inline void read(int &x) {
  13. x = 0; static char ch;
  14. while (!isdigit(ch = getchar())) ;
  15. while (x = x * 10 + (ch & 15), isdigit(ch = getchar()));
  16. }
  17. inline void write(LL x) {
  18. static char st[15];
  19. static char *t = st;
  20. while (x) *t++ = (x % 10) | 48, x /= 10;
  21. while (t != st) putchar(*--t);
  22. putchar('\n');
  23. }
  24. int N, K, L;
  25. int A[MN], B[MN];
  26. bool vA[MN], vB[MN];
  27. PQ A0, B0, A1, B1, C;
  28. LL Ans;
  29. int main() {
  30. freopen("sequence.in", "r", stdin);
  31. freopen("sequence.out", "w", stdout);
  32. int T; scanf("%d", &T);
  33. while (T--) {
  34. Ans = 0;
  35. while (!A0.empty()) A0.pop();
  36. while (!B0.empty()) B0.pop();
  37. while (!A1.empty()) A1.pop();
  38. while (!B1.empty()) B1.pop();
  39. while (!C.empty()) C.pop();
  40. read(N), read(K), read(L);
  41. for (int i = 1; i <= N; ++i) read(A[i]);
  42. for (int i = 1; i <= N; ++i) read(B[i]);
  43. for (int i = 1; i <= N; ++i) {
  44. vA[i] = vB[i] = 0;
  45. A0.push(mp(A[i], i));
  46. B0.push(mp(B[i], i));
  47. C.push(mp(A[i] + B[i], i));
  48. }
  49. int left = K - L;
  50. for (int cnt = 1; cnt <= K; ++cnt) {
  51. while (vA[A0.top().se]) A0.pop();
  52. while (vB[B0.top().se]) B0.pop();
  53. if (left) {
  54. pii a = A0.top(), b = B0.top();
  55. A0.pop(), B0.pop();
  56. Ans += a.fi + b.fi;
  57. int ap = a.se, bp = b.se;
  58. vA[ap] = vB[bp] = 1;
  59. if (ap == bp) C.pop();
  60. else {
  61. --left;
  62. if (vB[ap]) ++left, A1.pop();
  63. else B1.push(mp(B[ap], ap));
  64. if (vA[bp]) ++left, B1.pop();
  65. else A1.push(mp(A[bp], bp));
  66. }
  67. }
  68. else {
  69. pii c0 = C.empty() ? mp(0, 0) : C.top();
  70. while (vA[c0.se] || vB[c0.se])
  71. C.pop(), c0 = C.empty() ? mp(0, 0) : C.top();
  72. pii a0 = A0.top(), b0 = B0.top();
  73. pii a1 = A1.empty() ? mp(-b0.fi, 0) : A1.top();
  74. pii b1 = B1.empty() ? mp(-a0.fi, 0) : B1.top();
  75. int a = a1.fi + b0.fi;
  76. int b = b1.fi + a0.fi;
  77. if (c0.fi >= a && c0.fi >= b)
  78. Ans += c0.fi, vA[c0.se] = vB[c0.se] = 1;
  79. else if (a >= b) {
  80. Ans += a;
  81. A1.pop(), B0.pop(), vA[a1.se] = vB[b0.se] = 1;
  82. if (b1.se == b0.se) B1.pop(), ++left;
  83. else A1.push(mp(A[b0.se], b0.se));
  84. }
  85. else {
  86. Ans += b;
  87. B1.pop(), A0.pop(), vB[b1.se] = vA[a0.se] = 1;
  88. if (a1.se == a0.se) A1.pop(), ++left;
  89. else B1.push(mp(B[a0.se], a0.se));
  90. }
  91. }
  92. } write(Ans);
  93. }
  94. return 0;
  95. }

证明?不要问我证明啊!其实仔细分析一下应该是能证出来的,再不行就先套个费用流模型。

LOJ 3158: 「NOI2019」序列的更多相关文章

  1. Loj #3059. 「HNOI2019」序列

    Loj #3059. 「HNOI2019」序列 给定一个长度为 \(n\) 的序列 \(A_1, \ldots , A_n\),以及 \(m\) 个操作,每个操作将一个 \(A_i\) 修改为 \(k ...

  2. loj #2051. 「HNOI2016」序列

    #2051. 「HNOI2016」序列 题目描述 给定长度为 n nn 的序列:a1,a2,⋯,an a_1, a_2, \cdots , a_na​1​​,a​2​​,⋯,a​n​​,记为 a[1: ...

  3. LOJ 3160: 「NOI2019」斗主地

    题目传送门:LOJ #3160. 简要题意: 有一个长度为 \(n\) 的序列 \(a\),初始时 \(a_i=i\) 或 \(a_i=i^2\),这取决于 \(\mathrm{type}\) 的值. ...

  4. LOJ #2183「SDOI2015」序列统计

    有好多好玩的知识点 LOJ 题意:在集合中选$ n$个元素(可重复选)使得乘积模$ m$为$ x$,求方案数对$ 1004535809$取模 $ n<=10^9,m<=8000且是质数,集 ...

  5. LOJ 3059 「HNOI2019」序列——贪心与前后缀的思路+线段树上二分

    题目:https://loj.ac/problem/3059 一段 A 选一个 B 的话, B 是这段 A 的平均值.因为 \( \sum (A_i-B)^2 = \sum A_i^2 - 2*B \ ...

  6. LOJ 3159: 「NOI2019」弹跳

    题目传送门:LOJ #3159. 题意简述: 二维平面上有 \(n\) 个整点,给定每个整点的坐标 \((x_i,y_i)\). 有 \(m\) 种边,第 \(i\) 种边从 \(p_i\) 号点连向 ...

  7. LOJ 3156: 「NOI2019」回家路线

    题目传送门:LOJ #3156. 题意简述: 有一张 \(n\) 个点 \(m\) 条边的有向图,边有两个权值 \(p_i\) 和 \(q_i\)(\(p_i<q_i\))表示若 \(p_i\) ...

  8. 「NOI2019」序列

    NKOJ卡常卡不过QAQ description 给两个A,B序列,让你分别在A,B中各选k个数,其中至少有L对下标相等. Solution 把问题转化为至多选n-K对下标不同的对. 配对问题就用费用 ...

  9. loj#2002. 「SDOI2017」序列计数(dp 矩阵乘法)

    题意 题目链接 Sol 质数的限制并没有什么卵用,直接容斥一下:答案 = 忽略质数总的方案 - 没有质数的方案 那么直接dp,设\(f[i][j]\)表示到第i个位置,当前和为j的方案数 \(f[i ...

随机推荐

  1. [LeetCode] 911. Online Election 在线选举

    In an election, the i-th vote was cast for persons[i] at time times[i]. Now, we would like to implem ...

  2. Spring 常犯的十大错误,这坑你踩过吗?

    阅读本文大概需要 9 分钟. 1.错误一:太过关注底层 我们正在解决这个常见错误,是因为 “非我所创” 综合症在软件开发领域很是常见.症状包括经常重写一些常见的代码,很多开发人员都有这种症状. 虽然理 ...

  3. LInux 学习笔记系列

    1.Linux 就该这么学的笔记系列

  4. 排列组合(包括n中随机抽取m个)

    有些情况我们需要获取一个数组中的所有排列组合情况,或者获取一部分数据进行随机组合,这个在python中有一个模块可以实现.具体情况如下 :::::::::::::::::::::::::::::::: ...

  5. C++:= delete

    = delete delete的由来 如之前提到的,在没有声明默认特殊成员函数的时候,编译器会自动帮我们补充,但有时候我们并不希望存在这些函数,比如:我们不希望某个类通过拷贝的方式实例化一个新的对象. ...

  6. Unity调用windows系统dialog 选择文件夹

    #region 调用windows系统dialog 选择文件夹 [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)] public ...

  7. Phaser也可以实现countdownLatch的功能

    /** * 可用用phaser模拟countDownLatch * awaitAdvance方法:如果传入的参数和当前的phase相等,线程就阻塞住等待phase的值增加:否则就立即返回 */ pub ...

  8. 【JVM学习】3.深入解析强引用、软引用、弱引用、幻象引用

    来源:公众号:猿人谷 关于强引用.软引用.弱引用.幻象引用的区别,在很多公司的面试题中经常出现,可能有些小伙伴觉得这个知识点比较冷门,但其实大家在开发中经常用到,如new一个对象的时候就是强引用的应用 ...

  9. 运维开发实践——基于Sentry搭建错误日志监控系统

    错误日志监控也可称为业务逻辑监控, 旨在对业务系统运行过程中产生的错误日志进行收集归纳和监控告警.似乎有那么点曾相识?没错... 就是提到的“APM应用性能监控”.但它又与APM不同,APM系统主要注 ...

  10. logger.error打印完整的错误堆栈信息

    使用Spring Boot项目中的日志打印功能的时候,发现调用Logger.errror()方法的时候不能完全地打印出网站的错误堆栈信息,只能打印出这个错误是一个什么错误. 为什么呢,原因在于这个方法 ...