B - K-th Number HDU - 6231

题目链接
题解:就是在 ≥ \geq len的[l,r]内的找出第k大的数,然后放到另一个数组中,然后在新数组中找到第M大的数。

题解: 我们在这里期望找到所有能进到数组的数,并且知道他们的数量后就可以解决这个问题,但是很明显这是超时的,所以我们可以用二分来解决这个问题,具体的就是我们找到 > > >某一个数的数量然后不断的更新即可。

下面是AC代码:

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std;
#define int long long
const int N=1e5+10;
int t,a[N],b[N],n,k,m;
int check(int x)
{
    int cnt=0;//大于x的数量
    int pos=0;//数组的指针
    int cur=0;
    int p=0;
    //首先定一个方向,从i开始,往后找
    for(int i=1;i<=n;i++)
    {
        while(pos<=n&&cnt<k) 
        {
            if(a[++pos]>=x) cnt++;
        }
        cur+=(n-pos+1);//数量
        if(a[i]>=x) cnt--; 
    }
    return cur;
}
signed main()
{
    int t;
    cin>>t;
    while(t--)
    {
        cin>>n>>k>>m;
        for(int i=1;i<=n;i++) cin>>a[i],b[i]=a[i];
        sort(b+1,b+1+n);
        int l=1,r=n,ans=0;
        while(l<=r)
        {
            int mid=l+r>>1;
            int x=check(b[mid]);
            if(x==m)
            {
                 ans=mid;
                 break;
            }
            else if(x>m)//还可以更大点
            {
                ans=mid;
                l=mid+1;
            }
            else r=mid-1;
        }
        cout<<b[ans]<<endl;
    }
    return 0;
}

现在解释一下这个二分为什么要这样,因为我们假设找到的二分数量恰好是m,那这个就是答案,我们直接记录输出。

然后我们就不断的缩小范围,假设最后缩小只剩下两个数了——3 4。

假设3满足大于的情况我们还期望往更大的找,这是就找到了4,而这个4就不满足情况了,但是上一步我用ans记录了,所以就能找到答案。但是如果没有记录,直接用l或r表示,最终l,r就到了4,最终输出4,所以我们要用ans记录一下。

但是如果这个4满足题意的话,我们要输出4,如果不加=的话,我们到了4就停止了,但是还没有判断这个4符不符合条件,所以我们加上=,而加上=的情况下,用这种方法也一定可以跳出循环。

普通的二分答案就是l=mid,r=mid+1这种最终找到的l==r,这种情况下一定是满足终态一定时符合题意的,但是这种情况下需要判断。

总之,二分在做的时候多试一试就行。

总结:这个题目用到的知识点时尺取:顾名思义,就是用尺子量取一段,然后不断的推进左右边界。整体下可以用O(n)的复杂度。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值