题目描述:

给定两个整型数组A和B。我们将A和B中的元素两两相加可以得到数组C。
譬如A为[1,2],B为[3,4].那么由A和B中的元素两两相加得到的数组C为[4,5,5,6]。
现在给你数组A和B,求由A和B两两相加得到的数组C中,第K小的数字。

输入:

输入可能包含多个测试案例。
对于每个测试案例,输入的第一行为三个整数m,n, k(1<=m,n<=100000, 1<= k <= n *m):n,m代表将要输入数组A和B的长度。
紧接着两行, 分别有m和n个数, 代表数组A和B中的元素。数组元素范围为[0,1e9]。

输出:

对应每个测试案例,
输出由A和B中元素两两相加得到的数组c中第K小的数字。

样例输入:
2 2 3
1 2
3 4
3 3 4
1 2 7
3 4 5
样例输出:
5
6 此题真的很难。这个题的难度在于数组的长度非常长长长长长长长长长长长长长长长长长长长长,我开始的代码是这样的:
 #include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <algorithm>
#define MAX 100009 long long A[MAX];
long long B[MAX];
long long C[MAX]; int cmp(const void *a, const void *b) {
long long at = *(long long *)a;
long long bt = *(long long *)b;
return (int)(at - bt);
} int main(int argc, char const *argv[])
{
int m, n, k;
while(scanf("%d %d %d",&m, &n, &k) != EOF) {
for(int i = ; i < m; i++) {
scanf("%lld",&A[i]);
}
for(int i = ; i < n; i++) {
scanf("%lld",&B[i]);
}
int t = ;
for(int i = ; i < m; i++) {
for(int j = ; j < n; j++) {
C[t] = A[i] + B[j];
t++;
}
} qsort(C, t, sizeof(long long), cmp);
printf("%lld\n", C[k-]);
}
return ;
}

结果就run time error了

之后才明白此题需要用二分搜索,现对A,B两个数组排序,可知道他们的最大值,最小值。那么所要求的第几小的数字必然在这两个值之间,我们此时采用二分搜索的办法去找到这个值。具体的做法是先求出中间值mid,在求出中间值mid在序列中是第几小的数字,从而改变max值和min值,再不断的求下去。但一开始还是超时了,

 #include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <algorithm>
#define MAX 100009 long long A[MAX];
long long B[MAX]; int cmp(const void *a, const void *b) {
long long at = *(long long *)a;
long long bt = *(long long *)b;
return (int)(at - bt);
} long long findbit(long long mid, long long n, long long m) {
int ans = n*m;
for(long long int i = ; i < n; i++) {
for(long long int j = m -; j >= ; j--) {
if(A[i] + B[j] > mid) {
ans--;
}
else {
break;
}
}
}
return ans;
} int main(int argc, char const *argv[])
{
long long int m, n, k;
//freopen("input.txt","r",stdin);
while(scanf("%lld %lld %lld",&m, &n, &k) != EOF) {
for(int i = ; i < m; i++) {
scanf("%lld",&A[i]);
}
for(int i = ; i < n; i++) {
scanf("%lld",&B[i]);
} qsort(A, n, sizeof(long long),cmp);
qsort(B, m, sizeof(long long),cmp); long long max = A[n - ] + B[m - ];
long long min = A[] + B[];
long long mid; while(min <= max) {
mid = (max + min)/;
long long midBit = findbit(mid, n, m);
if(k <= midBit) {
max = mid - ;
}
else {
min = mid + ;
}
} printf("%lld\n",min);
}
return ;
}

超时的原因在于findBit内的二层循环。我们如果将第一个数组从小到大遍历,而将第二个数组从大到小遍历,比如下面

6 9 11

8 11 13

如果a[0] + b[1] > 15, (6 +11 > 15),那么a[1] + b[1] 必然也大于15,因为a[1] > a[0], a[1] + b[2]也必然大于15,因为b[2] > b[1]

所以需要遍历每一个a,而对于b只需要进行局部的遍历。

也就是说,对于a[0]生成的序列,mid是第mid0 = j+1小的数字,那么对于a[1], j只需要从上一个j开始算起,因为前面的肯定比mid要大。

代码如下

 #include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <algorithm>
#define MAX 100009 long long A[MAX];
long long B[MAX]; int cmp(const void *a, const void *b) {
long long at = *(long long *)a;
long long bt = *(long long *)b;
return (int)(at - bt);
} long long findbit(long long mid, long long n, long long m) {
long long int ans = ; long long int j = m - ;
for(long long int i = ; i < n; i++) {
while(j >= && A[i] + B[j] > mid) {
j--;
}
ans = ans + j + ;
}
return ans;
} int main(int argc, char const *argv[])
{
long long int m, n, k;
//freopen("input.txt","r",stdin);
while(scanf("%lld %lld %lld",&m, &n, &k) != EOF) {
for(int i = ; i < m; i++) {
scanf("%lld",&A[i]);
}
for(int i = ; i < n; i++) {
scanf("%lld",&B[i]);
} qsort(A, n, sizeof(long long),cmp);
qsort(B, m, sizeof(long long),cmp); long long max = A[n - ] + B[m - ];
long long min = A[] + B[];
long long mid; while(min <= max) {
mid = (max + min)/;
long long midBit = findbit(mid, n, m);
if(k <= midBit) {
max = mid - ;
}
else {
min = mid + ;
}
} printf("%lld\n",min);
}
return ;
}

九度oj 题目1534:数组中第K小的数字的更多相关文章

  1. 九度OJ 1534 数组中第K小的数字 -- 二分查找

    题目地址:http://ac.jobdu.com/problem.php?pid=1534 题目描述: 给定两个整型数组A和B.我们将A和B中的元素两两相加可以得到数组C. 譬如A为[1,2],B为[ ...

  2. 数组中第K小的数字(Google面试题)

    http://ac.jobdu.com/problem.php?pid=1534 题目1534:数组中第K小的数字 时间限制:2 秒 内存限制:128 兆 特殊判题:否 提交:1120 解决:208 ...

  3. 九度OJ 题目1534:数组中第K小的数字(二分解)

    题目链接:点击打开链接 题目描述: 给定两个整型数组A和B.我们将A和B中的元素两两相加可以得到数组C. 譬如A为[1,2],B为[3,4].那么由A和B中的元素两两相加得到的数组C为[4,5,5,6 ...

  4. 九度 1534:数组中第K小的数字(二分法变形)

    题目描述: 给定两个整型数组A和B.我们将A和B中的元素两两相加可以得到数组C.譬如A为[1,2],B为[3,4].那么由A和B中的元素两两相加得到的数组C为[4,5,5,6].现在给你数组A和B,求 ...

  5. 题目1534:数组中第K小的数字 ——二分

    http://ac.jobdu.com/problem.php?pid=1534 给定两个整型数组A和B.我们将A和B中的元素两两相加可以得到数组C.譬如A为[1,2],B为[3,4].那么由A和B中 ...

  6. 九度OJ 1351:数组中只出现一次的数字 (位运算)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:3098 解决:906 题目描述: 一个整型数组里除了两个数字之外,其他的数字都出现了两次.请写程序找出这两个只出现一次的数字. 输入: 每个 ...

  7. 九度OJ 1348:数组中的逆序对 (排序、归并排序)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:2777 解决:656 题目描述: 在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组 ...

  8. 九度OJ 题目1371:最小的K个数

    题目描述: 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 输入: 每个测试案例包括2行: 第一行为2个整数n,k(1< ...

  9. 九度OJ 题目1384:二维数组中的查找

    /********************************* * 日期:2013-10-11 * 作者:SJF0115 * 题号: 九度OJ 题目1384:二维数组中的查找 * 来源:http ...

随机推荐

  1. uvm_agent——007(特工)

    詹姆斯·邦德作为007的代言人,很好地诠释了agent的含义.但是在计算机系统中agent(代理)指能自主活动的软件或者硬件实体.在UVC中agent作为容器,实例化VIP的所有模块包括driver, ...

  2. winform中显示标题,点击打开链接

    效果:显示的是标题,但是点击打开的是链接 思路:定义一个类,将类实例化,向类中写入数据,再将类放到listbox中,设置listbox的显示分类为文本 前台:放入一个listbox控件 后台: pub ...

  3. mysql ERROR 2002 (HY000): Can't connect to local MySQL server through socket '/var/run/mysqld/mysqld.sock' (2 "No such file or directory")

    解决方案如下:

  4. 11gR2新特性---Gpnp守护进程

    在这篇文章中,我们会对11gR2 新的守护进程(资源名称ora.gpnpd)进行介绍,其中包含的gpnp的功能,启动顺序和基本的诊断方法. gpnp全称为grid plug and play,该组件的 ...

  5. WPF中窗体调用窗体

    在WPF中有时候我们需要在一个窗体中去调用另外的一个窗体,下面给出调用方法. 下面实现在MainWindow中通过点击一个按钮调用另外的一个窗口. 首先创建你要调用的另外一个窗口:点击最上面的项目  ...

  6. 使用js将后台返回的数据转换成树形结构

    将类似如下数据转换成树形的数据: [ { id: 1, name: '1', }, { id: 2, name: '1-1', parentId: 1 }, { id: 3, name: '1-1-1 ...

  7. webpack打包性能分析

    1. 如何定位webpack打包速度慢的原因 首先需要定位webpack打包速度慢的原因,才能因地制宜采取合适的方案,我们可以在终端输入: webpack --profile --json > ...

  8. 量化投资,你需要了解的A股财务数据

    摘要:基本面量化是应用量化研究领域的重头戏,财务数据的整理和加工是基本面量化的第一步.本文梳理了财务数据的基本知识,包括报表类型.数据来源.调整更正和使用原则等,并给出了单季度和TTM数据的计算流程. ...

  9. js 两个数组对象根据账号比较去重,解决直接splice后数组索引改变

    目的获取Arr2中不包含在arr1中的对象 根据Account进行比较,如果相等则删除tempArr数组对象. 结果返回张三 var arr1=[{"account":" ...

  10. poi导出word模板项目实例(一个文件)

    在页面上填写值,然后导出到word模板中,并把页面上的值带到模板中,也就是导出word文档,提前有word 的模板形式, 1.jsp 页面   <table class="formTa ...