原题链接

http://codevs.cn/problem/1245/

题目描述 Description

有两个长度为 N 的序列 A 和 B,在 A 和 B 中各任取一个数可以得到 N^2 个和,求这N^2 个和中最小的 N个。

输入描述 Input Description

第一行输入一个正整数N;第二行N个整数Ai 且Ai≤10^9;第三行N个整数Bi,

且Bi≤10^9

输出描述 Output Description

输出仅一行,包含 n 个整数,从小到大输出这 N个最小的和,相邻数字之间用

空格隔开。

样例输入 Sample Input

5

1 3 2 4 5 

6 3 4 1 7

样例输出 Sample Output

2 3 4 4 5

数据范围及提示 Data Size & Hint

【数据规模】 对于 100%的数据,满足 1≤N≤100000。

最简单的想法莫过于直接模拟,但是复杂度是O(n^2),不满足题目的数据范围的要求。我的解法是:

首先将数组A和B从小到大排序,令数组Sum[i]=A[i]+B[i]。

最开始的时候寻找满足0<=A[j]+B[k]<Sum[0]的j、k,将得到的和加入堆中,接下来寻找满足Sum[0]<=A[j]+B[k]<Sum[1]的j、k,将得到的和加入堆中。。。反复如此,直到堆的大小大于n。这样就将问题分为一段一段的求解。

现在的问题就在于如何寻找满足上述条件的j、k。

建一个数组tmp,对于A[j],当B[k]+A[j]>=Sum[i]的时候,tmp[j]=k并跳出。换句话说,tmp[j]表示A[j]在上一段的寻找中在B中停下的位置。

就拿样例来解释:

首先先对A,B排序:

A:1 2 3 4 5

B:1 3 4 6 7

然后求Sum:2 5 7 10 12

对于第一次,寻找小于等于2的A[j]+B[k],明显没有,所以直接讨论第二次,寻找小于5的A[j]+B[k],发现A[0]+B[2]>=5,所以tmp[0]=2,然后发现A[1]+B[1]>=5,所以tmp[1]=1,同理tmp[2]=1,tmp[3]=0,这时就没有继续在A上找下去的必要了。接下来要寻找小于7的A[j]+B[k],对于每一个A[j],都从B中tmp[j]中的位置开始寻找。复杂度是比O(n)稍大了一点,有个特殊情况是,Sum中的数个个相等,解决的办法是在A,B的末尾加一个很大的数。

详细见代码

  1. #include<iostream>
  2. #include<cstring>
  3. #include<queue>
  4. #include<vector>
  5. #include<cstdio>
  6. #include<functional>
  7. #include<algorithm>
  8. #define MAX_N 100005
  9. using namespace std;
  10.  
  11. typedef long long ll;
  12.  
  13. priority_queue<ll,vector<ll>,greater<ll> > que;
  14. int n;
  15. ll a[MAX_N],b[MAX_N];
  16. ll sum[MAX_N],tmp[MAX_N];
  17.  
  18. int main()
  19. {
  20. memset(sum,0,sizeof(sum));
  21. memset(tmp,0,sizeof(tmp));
  22. scanf("%lld ",&n);
  23. for(int i=0;i<n;i++)scanf("%lld",&a[i]);
  24. for(int i=0;i<n;i++)scanf("%lld",&b[i]);
  25.  
  26. sort(a,a+n);sort(b,b+n);
  27. a[n]=a[n-1]+100;b[n]=b[n-1]+100;n++;
  28. for(int i=0;i<n;i++)sum[i]=a[i]+b[i];
  29.  
  30. for(int i=0;i<n;i++)
  31. {
  32. ll s=sum[i];
  33. for(int j=0;j<n;j++)
  34. {
  35. int t=tmp[j];
  36. for(int k=t;k<n;k++)
  37. {
  38. ll sumTemp=a[j]+b[k];
  39. if(sumTemp>=s){tmp[j]=k;break;}
  40. que.push(sumTemp);
  41. }
  42. if(tmp[j]==0)break;
  43. }
  44. if(que.size()>=n)break;
  45. }
  46.  
  47. for(int i=0;i<n-1;i++)
  48. {
  49. printf("%lld ",que.top());
  50. que.pop();
  51. }
  52. cout<<endl;
  53. return 0;
  54. }

CODEVS 1245 最小的N个和 堆+排序的更多相关文章

  1. codevs 1245 最小的N个和

    1245 最小的N个和 http://codevs.cn/problem/1245/ 题目描述 Description 有两个长度为 N 的序列 A 和 B,在 A 和 B 中各任取一个数可以得到 N ...

  2. AC日记——最小的N个和 codevs 1245

    1245 最小的N个和  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题解  查看运行结果     题目描述 Description 有两个长度为 N ...

  3. 最小的N个和(codevs 1245)

    1245 最小的N个和  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题解  查看运行结果     题目描述 Description 有两个长度为 N ...

  4. 1245 最小的N个和

    1245 最小的N个和 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond         题目描述 Description 有两个长度为 N 的序列 A 和 B, ...

  5. 1245 最小的N个和(前k小ai+bi)

    1245 最小的N个和  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题解  查看运行结果     题目描述 Description 有两个长度为 N ...

  6. code vs 1245 最小的N个和

    1245 最小的N个和  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题解       题目描述 Description 有两个长度为 N 的序列 A ...

  7. Java实现 LeetCode 786 第 K 个最小的素数分数(大小堆)

    786. 第 K 个最小的素数分数 一个已排序好的表 A,其包含 1 和其他一些素数. 当列表中的每一个 p<q 时,我们可以构造一个分数 p/q . 那么第 k 个最小的分数是多少呢? 以整数 ...

  8. php实现把数组排成最小的数(核心是排序)(看别人的代码其实也没那么难)(把php代码也看一下)(implode("",$numbers);)(usort)

    php实现把数组排成最小的数(核心是排序)(看别人的代码其实也没那么难)(把php代码也看一下)(implode("",$numbers);)(usort) 一.总结 核心是排序 ...

  9. Codevs No.1245 最小的N个和

    2016-05-31 18:52:15 题目链接: 最小的N个和 Codevs No.1245 题目大意: 给两个等长数列,各取一个数求和,找到最小的N组 解法: 堆优化的大暴力 直接枚举所有可能在最 ...

随机推荐

  1. shell脚本,一个字符一个字符输出。

    [root@localhost wyb]# cat file abc def abc 789de f567 [root@localhost wyb]# cat fffile.sh #!/bin/bas ...

  2. Respond.js-----20150415

    Respond.js让IE6-8支持CSS3 Media Query. Bootstrap里面就引入了这个js文件,从名字看出来是自适应的兼容.打开IE看了一下,效果挺好的,自适应的效果挺好的.Res ...

  3. Luogu P2664 树上游戏 dfs+树上统计

    题目: P2664 树上游戏 分析: 本来是练习点分治的时候看到了这道题.无意中发现题解中有一种方法可以O(N)解决这道题,就去膜拜了一下. 这个方法是,假如对于某一种颜色,将所有这种颜色的点全部删去 ...

  4. PHP+Mysql实现分页

    我们在项目开发的过程中避免不了使用分页功能,拿php来说,现在市面上有很多大大小小的php框架,当然了分页这种小功能这些框架中都是拿来直接可以用的. 这些框架的分页功能使用都很方便,配置一下分页所需参 ...

  5. Makefile文件中的sed介绍

    haoxin$ sed --helpUsage: sed [OPTION]... {script-only-if-no-other-script} [input-file]... -n, --quie ...

  6. Luogu 2216 [HAOI2007]理想的正方形 (单调队列优化)

    题意: 给出一个 N×M 的矩阵,以及一个数值 K ,求在给定的矩阵中取出一个 K×K 的矩阵其中最大值减去最小值的最小值. 细节: 没有细节来发暴力走天下,20分也是分啊~~~ QAQ. 分析: 感 ...

  7. 打印 Python 的一切 —— pprint & beeprint

    打印,是所有程序员从小白时期就具备的神技,遇事不决打印一下,是 DEBUG 最简单且不依赖 IDE 的方式,自定义各种日志输出,也是项目成型后必备功能.但是为了优雅的打印格式,往往需要对各种对象进行特 ...

  8. LDAP学习小结【仅原理和基础篇】

    此篇文章花费了好几个晚上,大部分是软件翻译的英文文档,加上自己的理解所写,希望学习者能尊重每个人的努力. 我有句话想送给每个看我文章的人: 慢就是快,快就是慢!!! 另外更希望更多人能从认真从原理学习 ...

  9. luogu3834 【模板】可持久化线段树 1(主席树)

    关于空间,第零棵树是 \(4n\),其后每棵树都要多来 \(\log(n)\) 的空间,所以我是开 \(n(4+\log(n))\) 的空间. 关于借用节点: 图片来自这里 #include < ...

  10. 一步一步在ubuntu上安装即时通讯服务器-Openfire

    1.首先登录到ubuntu server.在安装openfire 服务器之前,先确保你的系统已经更新到最新.然后输入下面的命令,一行一行执行,最后安装可用的更新 sudo apt-get update ...