题目链接
题解:就是在
≥
\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)的复杂度。