51nod 1105---二分套二分

http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1105


题意a序列和b序列,ab序列是 a和b两两组合,问你ab中第k大是多少。。


这题是个二分套二分,是个二分好题。。为什么好呢,因为这个二分容易写残疾啊。。。(总之就是我太弱了

接下来分析一下这个题的解法和要注意的地方。

一个显然的方法就是二分答案了,然后判断mid是第几大就可以了。。

但是,仔细一想?咦。。会不会二分的答案其实不是ab数组里面的呢?会不会ab里面有很多个一样的值我无法返回正确答案呢?


我们先分析二分时候我们计算的是什么。。

我们计算的mid在ab数组里面时候,枚举a,二分b就可以知道有几个大于等于mid了,


在设计二分的时候-当返回值大于等于k的时候,使得l=mid+1;(参考blog--《你真的会二分查找吗?》)


我们这时候考虑一下我们会遇到的问题,

Q 计算的时候会不会有什么问题?例如mid不是ab数组里面的一个数。

A 我们先不考虑是不是ab数组里的一个数,我们计算的是比mid大的ab中的数有多少个。


Q     如果一个数是第k大但是这个数有很多呢?

A     (参考blog--《你真的会二分查找吗?》),这里返回第一个大于等于它的值。


会不会返回一个不在ab数组里的一个数呢?

A 答案是不会的,因为 这个问题可以划归到上一个问题,既然返回的是第一个大于等于它的值,那么如果mid不合法,那么一定存在一个和mid计算值一样的,并且这个合 法值要小于mid,(因为不存在的时候要向上取整的,自然虽然计算值一样但是存在的较小)。。


PS。。很多地方可能说的不对,望大家指正。


放代码:

#include <iostream>
#include <algorithm>
#include <vector>
using namespace std;

const int maxn=50000;
long long n,k;
long long A[maxn],B[maxn];


long long Ca(long long x){
    long long f;
    long long ret1=0;
    for(int i=0;i<n;i++){
        f=x/A[i];
        if(x%A[i]!=0)
            f++;
        if(B[n-1]>=f)
            ret1+=(n-(long long)(lower_bound(B,B+n,f)-B));
    }
    return ret1;
}

int main()
{
    cin>>n>>k;
    for(int i=0;i<n;i++){
        cin>>A[i]>>B[i];
    }
    sort(A,A+n);
    sort(B,B+n);
    long long l=A[0]*B[0],r=A[n-1]*B[n-1],mid;
    while(l<=r){
        mid=(l+r)/2;
        long long x=Ca(mid);
        if(x>=k){
            l=mid+1;
        }
        else{
            r=mid-1;
        }
    }
    cout<<l-1<<endl;
    return 0;
}


你可能感兴趣的:(51nod 1105---二分套二分)