题目描述:

给定两个整型数组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. 浅析linux下软件的安装

    Linux环境: CentOs 6.0 知识点介绍: 一.tarball安装 安装步骤: 将tarball文件在/usr/local/src目录解压缩 ./configure:这个步骤是建立makef ...

  2. jmeter中登录和提交收银出现的错误

    登录出现的错误 登录界面如图所示: 为了防止登录跳转的问题response code 302的问题,要设置 2.提交收银界面 当系统设置必须传送jison格式时,要在HTTP Header Manag ...

  3. hadoop中修改端口号

    1.hdfs-site.xml 这里修改hdfs相关的端口. 1 <property> 2 <name>dfs.namenode.scondary.http-address&l ...

  4. App Transport Security has blocked a cleartext HTTP

    打开info.plist源代码复制粘贴     <key>NSAppTransportSecurity</key>     <dict> <!--Includ ...

  5. android-menudrawer-master 使用

    1. 参照例子写, 运行总崩溃, 多半是导库问题... 2. 既然这样不行, 只好将源码全部拷贝到工程中了. 然后修错误, (将需要的res 文件复制过来.主要是value中的几个文件) 3. 在Ma ...

  6. SpringAOP 设计原理

    1.  设计原理 引入了,代理模式. java 程序执行流: 如果从虚拟机的角度看,整个程序的过程就是方法的调用,我们按照方法的执行顺序,将方法调用成一串. 在方法之间有着Join Point 连接点 ...

  7. Kubernetes介绍与特性

    1.Kubernetes 是什么 简单的来说,k8s可以理解为,一个容器平台,一个微服务平台,便携式云平台,我们那可以很快速的搭建一个服务,快速的运行起来 2.Kubernetes特性

  8. Vue的elementUI实现自定义主题

    使用vue开发项目,用到elementUI,根据官网的写法,我们可以自定义主题来适应我们的项目要求,下面来介绍一下两种方法实现的具体步骤,(可以参考官方文档自定义主题官方文档),先说项目中没有使用sc ...

  9. iOS利用UIDocumentInteractionController和Quick Look打开或预览文档

    在App的开发过程中,我们避免不了要打开软件中的文件,例如:Excel文件,Word文件,图片文件等不同格式的文件或者想要通过第三方的App来打开这些文件,那么我们就要用到UIDocumentInte ...

  10. ios之自定义导航栏上的返回按钮

    导航栏的按钮,右边的按钮是可以自己随意添加的.但左边的返回按钮怎么定制?你会说,添加一个自己的按钮呗!你可以试试看,这样行不行. 正确的答案是重载UINavigationController类的pus ...