当前位置:   article > 正文

【题解】ATCoder_abc123_D - Cake 123_atcoder 每次给数组中的一个元素加一,动态维护数组中的最大值

atcoder 每次给数组中的一个元素加一,动态维护数组中的最大值

Link: D - Cake 123

Description:

现有3种蛋糕,每个蛋糕都有一个称为美味度的整数值,如下所示:

  • 第一种蛋糕的美味度是 A_1,A_2,\cdot \cdot \cdot ,A_X
  • 第二种蛋糕的美味度是 B_1,B_2,\cdot \cdot \cdot ,B_Y
  • 第三种蛋糕的美味度是 C_1,C_2,\cdot \cdot \cdot ,C_Z

现要求买三个蛋糕,每种蛋糕各一个,显然共有X\times Y\times Z种方法,对其按照美味度总和降序排列。输出前 K 种方法的蛋糕美味度总和。

Constraints:

  • 1 \leq X,Y,Z \leq 1000
  • 1 \leq K \leq min(3000,X\times Y\times Z)
  • 1\leq A_i,B_i,C_i \leq 10^{10}

Analysis_1: 

  • 无脑暴力枚举后排序,显然复杂度O(NlogN),N=X\times Y\times Z爆爆爆!,故考虑如何优化。分析数据规模,注意到K的范围最大才 3000,故可以尝试枚举前两种蛋糕的所有组合,取前 K 种,再用这 K 种进一步和第三种蛋糕去组合,再取前 K 种即可。
  • O(K^2logK)

Sulution_1:

  1. vector<ll> a,b,c;
  2. int main() {
  3. int x,y,z,K;
  4. cin >> x >> y >> z >> K;
  5. for(int i=0;i<x;i++) {
  6. ll t; cin >> t;
  7. a.push_back(t);
  8. }
  9. for(int i=0;i<y;i++) {
  10. ll t; cin >> t;
  11. b.push_back(t);
  12. }
  13. for(int i=0;i<z;i++) {
  14. ll t; cin >> t;
  15. c.push_back(t);
  16. }
  17. //处理前两种组合
  18. vector<ll> ab;
  19. for(int i=0;i<x;i++) {
  20. for(int j=0;j<y;j++) {
  21. ab.push_back(a[i]+b[j]);
  22. }
  23. }
  24. sort(ab.begin(),ab.end(),greater<ll>());
  25. // 取前K种和第三种组合,注意前两种若不够K,要取min
  26. vector<ll> ans;
  27. for(int i=0;i<min(K,x*y);i++) {
  28. for(int j=0;j<z;j++) {
  29. ans.push_back(ab[i]+c[j]);
  30. }
  31. }
  32. sort(ans.begin(),ans.end(),greater<ll>());
  33. for(int i=0;i<K;i++) cout << ans[i] << endl;
  34. return 0;
  35. }

Analysis_2: 

  • 如果把每种蛋糕分别排序呢?每次按美味度最大的依次取出,可以保证一定在前 K 种,但不一定是严格降序的,故对这些满足条件的再做一次排序即可。
  • O(Klog^3K)

Solution_2:

  1. ll a[1005],b[1005],c[1005];
  2. int main() {
  3. int x,y,z,K;
  4. cin >> x >> y >> z >> K;
  5. for(int i=0;i<x;i++) cin >> a[i];
  6. for(int i=0;i<y;i++) cin >> b[i];
  7. for(int i=0;i<z;i++) cin >> c[i];
  8. sort(a,a+x,greater<ll>());
  9. sort(b,b+y,greater<ll>());
  10. sort(c,c+z,greater<ll>());
  11. vector<ll> ans;
  12. for(int i=0;i<x;i++) {
  13. for(int j=0;j<y;j++) {
  14. for(int k=0;k<z;k++) {
  15. //按照序号依次取
  16. if((i+1)*(j+1)*(k+1) <= K) ans.push_back(a[i]+b[j]+c[k]);
  17. }
  18. }
  19. }
  20. sort(ans.begin(),ans.end(),greater<ll>());
  21. for(int i=0;i<K;i++) cout << ans[i] << endl;
  22. return 0;
  23. }

Analysis_3: 

  • 顺着上面的思路,(分别排好序后)和最大的肯定是 A_1+B_1+C_1,次大的一定是 A_2+B_1+C_1, A_1+B_2+C_1,A_1+B_1+C_2三者其一,即只将一个最大值改为次大值,以此类推,每次都面临相同的选择,故我们要动态维护这个最大值,自然可以想到用优先队列(priority_queue)。当前的最大值如果是 A_i+B_j+C_k,则就把上述的三个值丢到这个大根堆里,进行 K 次操作即可。
  • 此外,还需要考虑重复的情况,比如由 A_{i-1},B_j,C_k 可以产生 A_i,B_j,C_k,而 A_i,B_{j-1},C_k 也可以产生 A_i,B_j,C_k,故可用 map下标组合去重(而不是对和的值去重!!),具体细节注意一下可以用 pair套pair 来确定3个下标,当然也可以再开个结构体去重载小于号(懒得写有点麻烦....)
  • O(KlogK)

Solution_3:

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. typedef long long ll;
  4. typedef unsigned long long ull;
  5. ll a[1005],b[1005],c[1005];
  6. struct Node{
  7. int i,j,k;
  8. Node(){}
  9. Node(int i,int j,int k):i(i),j(j),k(k){}
  10. bool operator < (const Node &A) const {
  11. return a[i]+b[j]+c[k] < a[A.i]+b[A.j]+c[A.k];
  12. }
  13. };
  14. priority_queue<Node> pq;
  15. map<pair<pair<int,int>,int>,int> mp; //下标组合去重
  16. int main() {
  17. int x,y,z,K;
  18. cin >> x >> y >> z >> K;
  19. for(int i=0;i<x;i++) cin >> a[i];
  20. for(int i=0;i<y;i++) cin >> b[i];
  21. for(int i=0;i<z;i++) cin >> c[i];
  22. sort(a,a+x,greater<ll>());
  23. sort(b,b+y,greater<ll>());
  24. sort(c,c+z,greater<ll>());
  25. pq.push(Node(0,0,0));
  26. mp[make_pair(make_pair(0,0),0)] = 1;
  27. while(K--) {
  28. Node now = pq.top();
  29. int ni=now.i; int nj=now.j; int nk=now.k;
  30. pq.pop();
  31. cout << a[ni] + b[nj] + c[nk] << endl;
  32. // 下标合法 && 下标组合未重复
  33. if(ni+1 < x && !mp.count(make_pair(make_pair(ni+1,nj),nk))) {
  34. pq.push(Node(ni+1,nj,nk));
  35. mp[make_pair(make_pair(ni+1,nj),nk)] = 1;
  36. }
  37. if(nj+1 < y && !mp.count(make_pair(make_pair(ni,nj+1),nk))) {
  38. pq.push(Node(ni,nj+1,nk));
  39. mp[make_pair(make_pair(ni,nj+1),nk)] = 1;
  40. }
  41. if(nk+1 < z && !mp.count(make_pair(make_pair(ni,nj),nk+1))) {
  42. pq.push(Node(ni,nj,nk+1));
  43. mp[make_pair(make_pair(ni,nj),nk+1)] = 1;
  44. }
  45. }
  46. return 0;
  47. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/羊村懒王/article/detail/74442
推荐阅读
相关标签
  

闽ICP备14008679号