原题链接

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的末尾加一个很大的数。

详细见代码

#include<iostream>
#include<cstring>
#include<queue>
#include<vector>
#include<cstdio>
#include<functional>
#include<algorithm>
#define MAX_N 100005
using namespace std; typedef long long ll; priority_queue<ll,vector<ll>,greater<ll> > que;
int n;
ll a[MAX_N],b[MAX_N];
ll sum[MAX_N],tmp[MAX_N]; int main()
{
memset(sum,0,sizeof(sum));
memset(tmp,0,sizeof(tmp));
scanf("%lld ",&n);
for(int i=0;i<n;i++)scanf("%lld",&a[i]);
for(int i=0;i<n;i++)scanf("%lld",&b[i]); sort(a,a+n);sort(b,b+n);
a[n]=a[n-1]+100;b[n]=b[n-1]+100;n++;
for(int i=0;i<n;i++)sum[i]=a[i]+b[i]; for(int i=0;i<n;i++)
{
ll s=sum[i];
for(int j=0;j<n;j++)
{
int t=tmp[j];
for(int k=t;k<n;k++)
{
ll sumTemp=a[j]+b[k];
if(sumTemp>=s){tmp[j]=k;break;}
que.push(sumTemp);
}
if(tmp[j]==0)break;
}
if(que.size()>=n)break;
} for(int i=0;i<n-1;i++)
{
printf("%lld ",que.top());
que.pop();
}
cout<<endl;
return 0;
}

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. pre-receive hook declined

    没有master分支的代码提交权限. 分配权限或者提交分支合并申请

  2. C# 使用Epplus导出Excel [3]:合并列连续相同数据

    C# 使用Epplus导出Excel [1]:导出固定列数据 C# 使用Epplus导出Excel [2]:导出动态列数据 C# 使用Epplus导出Excel [3]:合并列连续相同数据 C# 使用 ...

  3. luoguP1164 小A点菜(背包问题)

    题目背景 uim神犇拿到了uoi的ra(镭牌)后,立刻拉着基友小A到了一家……餐馆,很低端的那种. uim指着墙上的价目表(太低级了没有菜单),说:“随便点”. 题目描述 不过uim由于买了一些辅(e ...

  4. 详解css媒体查询

    简介 媒体查询(Media Queries)早在在css2时代就存在,经过css3的洗礼后变得更加强大bootstrap的响应式特性就是从此而来的. 简单的来讲媒体查询是一种用于修饰css何时起作用的 ...

  5. centos7.2 安装nginx

    一.首先检查gcc编译器安装了没 二 首先安装必要的库(nginx 中gzip模块需要 zlib 库,rewrite模块需要 pcre 库,ssl 功能需要openssl库),检查一下是否已经安装了( ...

  6. C++ 字符串分割,并把子字符串转换成int型整数

    主要涉及到string类的两个函数find和substr: find()函数的用法: 原型:size_t find ( const string& str, size_t pos = 0 ) ...

  7. 03007_HttpServlet

    1.创建 new---Servlet package com.gzdlh.servlet; import java.io.IOException; import javax.servlet.Servl ...

  8. adb 命令大全

    传送门 --> https://github.com/mzlogin/awesome-adb ADB,即 Android Debug Bridge,它是 Android 开发/测试人员不可替代的 ...

  9. python基础-文件和目录

    字符串小练习 >>> s="1a2a3a4a5a" >>> s1=s.split('a') >>> >>> ...

  10. Leetcode 385.字典序排序

    字典序排序 给定一个整数 n, 返回从 1 到 n 的字典顺序. 例如, 给定 n =1 3,返回 [1,10,11,12,13,2,3,4,5,6,7,8,9] . 请尽可能的优化算法的时间复杂度和 ...