题目描述:

给定两个整型数组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. OutOfMemory

    查看图片格式,如果为PNG,可更改为jpg.图片会变小. 停止activity 当activity调用onStop()方法, activity不再可见,并且应该释放那些不再需要的所有资源.一旦acti ...

  2. COGS 1710. [POJ2406]字符串的幂

    ★☆   输入文件:powerstrings.in   输出文件:powerstrings.out   简单对比时间限制:3 s   内存限制:256 MB [题目描述] 对于给定的两个字符串a,b, ...

  3. gunzip

    gunzip——解压缩.gz的压缩文件 GNU unzip 命令所在路径:/bin/gunzip 示例1: # gunzip services.gz 解压缩当前目录下的services.gz文件,执行 ...

  4. 添加/删除 windows下Git右键菜单

    从网上搜索了些方法,总结一下 不显示右键菜单: 方法1: 安装的时候选择不添加右键菜单. 方法2(绝对有效): 运行CMD Windows 64 输入命令(地址为git安装地址) cd "C ...

  5. 解决android studio设置版本号

    获取版本号代码 /** * 获取版本号 * @return 当前应用的版本号 */ public static String getVersion(Context context) { try { P ...

  6. k8s学习目录

    目录 K8S基础部分 基础部分 5 秒创建 k8s 集群[转] k8s 核心功能[转] k8s 重要概念[转] 部署 k8s Cluster(上)[转] 部署 k8s Cluster(下)[转] Ku ...

  7. ES6新增Map、Set和iterable

    Map需要一个二维数组 var test_map = new Map(["mians",99],["regink",88]) test_map.get(&quo ...

  8. python爬虫---实现项目(四) 用BeautifulSoup分析新浪新闻数据

    这次只演示了,如何在真实项目内用到BeautifulSoup库来解析网页,而新浪的新闻是ajax加载过来的数据,在这里我们只演示解析部分数据(具体反扒机制没做分析). 代码地址:https://git ...

  9. Bootstrap历练实例:大的按钮

    <!DOCTYPE html><html><head> <meta http-equiv="Content-Type" content=& ...

  10. 安装pycharm 2018.3 Professional Edition

    1.下载pycharm 2018.3 Professional 2.下载破解补丁,Gitee仓库 或 直接下载(Direct download link) ,并放到pycharm目录下的\bin目录( ...