传送门

Description

Input

Output

Translation

·  给定k个长度为k的数组,把每个数组选一个元素加起来,这样共有kk种可能的答案,求最小的k个

Sample Input


Sample Output

 

Hint

k<=750

Solution

  显然可以一行一行做,同时如果S+now[j]最小,需要S最小。即我们只需要记录前i行的最小的k个ans,分别与当前行的k个相加,从k2个ans中选择前k个小的记录。显然前k小的可以开一个大根堆维护。即如果size>k则pop。这样一共有k次转移,每次转移有k2个状态,维护k的堆的复杂度为Logk,这样总复杂度为O(k3logk)

  易于通过数学归纳证明取第i行的最小的只需要与前i-1行最小的k个ans相加得到。这样我们不妨用数组ans记录前i行最小的k个ans,现在我们考虑我们有两个数组a,b,需要取k个ans最小。不妨设a和b都是有序的,我们显然有如下关系:

  a1+b1≤a1+b2≤a1+b3≤a1+b4≤……≤a1+bk

  a2+b1≤a2+b2≤a2+b3≤a2+b4≤……≤a2+bk

  ……

  ak+b1≤ak+b2≤ak+b3≤ak+b4≤……≤ak+bk

这样显然ai+bj可能成为前k小的ans当且仅当ai+bj-1是前k小的ans。

我们维护一个可能成为ans的序列,每次取这个序列中最小,显然最小值可以成为合法的ans。同时不妨设这个值为ai+bj,那么我们将ai+bj+1压入序列,因为它可能成为合法的ans。

初始化上,因为ai+b1可能成为合法的ans,我们将这k个全部压入队列中。

Code

#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 1010 inline void qr(int &x) {
char ch=getchar(),lst=NULL;
while(ch>''||ch<'') lst=ch,ch=getchar();
while(ch>=''&&ch<='') x=(x<<)+(x<<)+(ch^),ch=getchar();
if(lst=='-') x=-x;
} template <typename T>
inline T mmax(const T &a,const T &b) {if(a>b) return a;return b;}
template <typename T>
inline T mmin(const T &a,const T &b) {if(a<b) return a;return b;}
template <typename T>
inline T mabs(const T &a) {if(a>=) return a;return -a;} template <typename T>
inline void mswap(T &a,T &b) {T temp=a;a=b;b=temp;} int k,MU[maxn],ans[maxn]; struct Zay {
int v,s;
Zay(int a=,int b=) {v=a;s=b;}
inline bool operator <(const Zay &others) const{return this->v>others.v;}
}; std::priority_queue<Zay>Q;
void clear(); int main() {
while(~scanf("%d",&k)) {
clear();
for(int i=;i<=k;++i) qr(ans[i]);
for(int i=;i<=k;++i) {
memset(MU,,sizeof MU);
for(int j=;j<=k;++j) {
qr(MU[j]);
}
std::sort(MU+,MU++k);
while(!Q.empty()) Q.pop();
for(int j=;j<=k;++j) {
Q.push(Zay(ans[j]+MU[],));
}
for(int j=;j<=k;++j) {
Zay temp=Q.top();Q.pop();
ans[j]=temp.v;
if(temp.s<k) Q.push(Zay(temp.v-MU[temp.s]+MU[temp.s+],temp.s+));
}
}
for(int i=;i<k;++i) printf("%d ",ans[i]);printf("%d\n",ans[k]);
}
return ;
} void clear() {
memset(ans,,sizeof ans);
}

Summary

在堆的应用中,维护一坨可能合法的解进行操作,是一种常用的思路。比如本题和dijkstra算法都是这个思路。这一坨合法的解一般满足下面两个条件:

第一,这坨解中最大/小的解一定是合法的解。比如本题中,序列中最小的一定是合法的解,dijkstra算法中,堆中权值和最小的解一定是到该点的最短路等等

第二,通过寻找合法解,可以获得其他可能解。比如在本题中,ai+bj合法时,ai+bj+1是可能解。在dijkstra算法中,找到到一个点的最短路可以更新和它相邻的点的可能最短路压入堆中。

【优先队列】【UVa11997】K Smallest Sums的更多相关文章

  1. 【暑假】[实用数据结构]UVa11997 K Smallest Sums

    UVa11997 K Smallest Sums  题目: K Smallest Sums You're given k arrays, each array has k integers. Ther ...

  2. UVA-11997 K Smallest Sums

    UVA - 11997 K Smallest Sums Time Limit: 1000MS   Memory Limit: Unknown   64bit IO Format: %lld & ...

  3. uva11997 K Smallest Sums&&UVALive 3135 Argus(优先队列,多路归并)

    #include<iostream> #include<cstdio> #include<cstdlib> #include<cstring> #inc ...

  4. UVA11997 K Smallest Sums

    思路 经典的k路归并问题 问题先转换为2路的有序表归并 先让A[1~k]都和B[1]相加,然后加入堆中,取出堆顶(A[x]+B[y])之后,再放入A[x]+B[y+1] 代码 #include < ...

  5. 题解——UVA11997 K Smallest Sums

    题面 背景 输入 输出 翻译(渣自翻) 给定K个包含K个数字的表,要求将其能产生的\( k^{k} \)个值中最小的K个输出出来 题解 k路归并问题的经典问题 可以转化为二路归并问题求解 考虑A[], ...

  6. 11997 - K Smallest Sums(优先队列)

    11997 - K Smallest Sums You’re given k arrays, each array has k integers. There are kk ways to pick ...

  7. UVa 11997 K Smallest Sums 优先队列&amp;&amp;打有序表&amp;&amp;归并

    UVA - 11997 id=18702" target="_blank" style="color:blue; text-decoration:none&qu ...

  8. D - K Smallest Sums(多路归并+贪心)

    Problem K K Smallest Sums You're given k arrays, each array has k integers. There are kk ways to pic ...

  9. 优先队列 UVA 11997 K Smallest Sums

    题目传送门 题意:训练指南P189 分析:完全参考书上的思路,k^k的表弄成有序表: 表1:A1 + B1 <= A1 + B2 <= .... A1 + Bk 表2:A2 + B1 &l ...

随机推荐

  1. 在使用Pipeline串联多个stage时model和非model的区别

    train.csv数据: id,name,age,sex1,lyy,20,F2,rdd,20,M3,nyc,18,M4,mzy,10,M 数据读取: SparkSession spark = Spar ...

  2. Ubuntu16.04安装wps办公软件解决文字缺失

    先到wps官网下载linux版wps安装包 选择64位的alpha版本下载: 下载完后,同样是cd到Downloads目录,用dpkg命令来安装它: cd  Downloads/ sudo dpkg ...

  3. 【第三章】Shell 变量的数值计算

    一.算数运算符 shell中常见的算术运算符: shell中常见的算术命令: 1. 整数运算 方法一:expr  expr命令就既可以用于整数运算,也可以用于相关字符串长度.匹配等的运算处理: exp ...

  4. CryptoZombies学习笔记——Lesson5

    chapter1:token代币 简而言之,通证就是支持交易的包含一系列规范的函数接口的一个智能合约,发币可以用ERC20标准,但是像僵尸这种非同质化代币,需要用ERC721标准 chapter2:e ...

  5. Java 单例模式探讨

    以下是我再次研究单例(Java 单例模式缺点)时在网上收集的资料,相信你们看完就对单例完全掌握了 Java单例模式应该是看起来以及用起来简单的一种设计模式,但是就实现方式以及原理来说,也并不浅显哦. ...

  6. 【转载】java byte转十六进制

    public static String bytes2HexString(byte[] b) { String ret = ""; for (int i = 0; i < b ...

  7. JS判断备忘

    快速引入jquery并显示重点内容 (function(d,j,s,t){t=d.body.appendChild(d.createElement("script"));t.onl ...

  8. Java中ArrayList与数组间相互转换

    在实际的 Java 开发中,如何选择数据结构是一个非常重要的问题. 衡量标准化(读的效率与改的效率) : ① Array: 读快改慢 ② Linked :改快读慢 ③ Hash:介于两者之间 实现Li ...

  9. LintCode-381.螺旋矩阵 II

    螺旋矩阵 II 给你一个数n生成一个包含1-n^2的螺旋形矩阵 样例 n = 3 矩阵为 [     [ 1, 2, 3 ],     [ 8, 9, 4 ],     [ 7, 6, 5 ] ] 标 ...

  10. TCP系列27—窗口管理&流控—1、概述

    在前面的内容中我们依次介绍了TCP的连接建立和终止过程和TCP的各种重传方式.接着我们在这部分首先关注交互式应用TCP连接相关内容如延迟ACK.Nagle算法.Cork算法等,接着我们引入流控机制(f ...