AC代码
这题思路就是二分答案,二分可能的dis并检验
在复杂的就是check()函数必须在logn的级别,写的时候也比较复杂,这边简单阐述一下
1 2 3
| ll l=lower_bound(A.begin(),A.end(),obl)-A.begin(); ll r=upper_bound(A.begin(),A.end(),obr)-A.begin();
|
这个代码是用来找到该dis可以包含的最多点数,详见下面图解

b=2 , dis=3,包含的最多点数就是-1,-1,5,5,即4个点,在图上很清楚,upper_bound-lower_bound就是4。
然后我们知道,最多点≥k是可能满足条件的(可能最少点符合条件),但最多点<k,那么一定不符合条件
AC
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61
| #include <iostream> #include <cstring> #include <algorithm> #include <deque> #include <queue> #include <vector> #include <set> #include <map> #include <cmath> #include <bitset> #include <iterator> #include <random>
using namespace std; typedef long long ll; const ll N=static_cast<ll>(1e5)+10; ll n,q,rangeA; vector<ll> A;
inline bool check(ll b,ll dis,ll k) { ll obl=b-dis,obr=b+dis; ll l=lower_bound(A.begin(),A.end(),obl)-A.begin(); ll r=upper_bound(A.begin(),A.end(),obr)-A.begin(); if(r-l<k) return false; return true; }
inline ll bsearch(ll b,ll k) { ll l=0,r=rangeA; while (l<r) { ll mid= l+r >>1; if(check(b,mid,k)) r=mid; else l=mid+1; } return l; }
int main() { ios::sync_with_stdio(false); cin.tie(0);cout.tie(0); cin>>n>>q; for(int i=1;i<=n;i++) { ll a; cin>>a; A.push_back(a); } sort(A.begin(),A.end()); for(int i=1;i<=q;i++) { ll b,k; cin>>b>>k; rangeA=max(abs(b-A[0]),abs(b-A[n-1])); cout<<bsearch(b,k)<<'\n'; } }
|
心路历程(WA,TLE,MLE……)