题目:https://www.luogu.org/problemnew/show/P3957
先二分一个 g,然后判断;
由于转移的范围是一个区间,也就是滑动窗口,所以单调队列优化;
可以先令队尾为 -1,但不真的放进去,为的是第一次判断能否从0走到;
普及组的题也要 Narh 提点...
代码如下:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int const xn=5e5+;
int n,d,k,x[xn],s[xn],q[xn],h,t;
ll f[xn],inf=1e17;
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=(ret<<)+(ret<<)+ch-'',ch=getchar();
return f?ret:-ret;
}
void add(int x)
{
while(h<=t&&f[q[t]]<=f[x])t--;
q[++t]=x;
}
bool ck(int g)
{
h=,t=; int mn=max(d-g,);
memset(f,-,sizeof f);
x[]=; q[]=-; f[]=;
for(int i=;i<=n;i++)
{
while(x[i]-x[q[t]+]>=mn)add(q[t]+);//
while(h<=t&&x[i]-x[q[h]]>d+g)h++;
if(h<=t)f[i]=f[q[h]]+s[i];
if(f[i]>=k)return ;
}
return ;
}
int main()
{
n=rd(); d=rd(); k=rd();
for(int i=;i<=n;i++)x[i]=rd(),s[i]=rd();
int l=,r=max(x[n]-d,d-),ans=-;
while(l<=r)
{
int mid=((l+r)>>);
if(ck(mid))ans=mid,r=mid-;
else l=mid+;
}
printf("%d\n",ans);
return ;
}