ACM/ICPC 2018亚洲区预选赛北京赛站网络赛

时间:2021-07-03 17:12:27

题意:到一个城市得钱,离开要花钱。开始时有现金。城市是环形的,问从哪个开始,能在途中任意时刻金钱>=0;

一个开始指针i,一个结尾指针j。指示一个区间。如果符合条件++j,并将收益加入sum中(收益可能是负数)。不符合就++i,并从sum中退掉收益直到sum>=0;区间长度为n时i的位置就是结果。

正确性证明:假设[a,b]是第一个合法区间。某时刻i,j都<a,i在a左边,因此j不可能扩到b右边(否则[a,b]不是第一个合法区间)。只可能j仍落在a左边,或者落到a,b中间。在i<a时,j不可能扩到b右边。j在a,b中间时,i不可能被退到a右边。

这样的尺取法以往还用来找和大于某值的最短连续区间。

#include <iostream>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <vector>
#include <iomanip>
#include <cstring>
#include <map>
#include <queue>
#include <set>
#include <cassert>
#include <stack>
using namespace std;
const double EPS=1e-;
typedef long long lon;
const lon SZ=,INF=0x7FFFFFFF;
lon arr[*SZ],nex[SZ]; int main()
{
std::ios::sync_with_stdio();
//freopen("d:\\1.txt","r",stdin);
lon casenum;
cin>>casenum;
//scanf("%d",&casenum);
for(lon time=;time<=casenum;++time)
//for(lon time=1;scanf("%d%d",&n,&k)!=EOF;++time)
{
lon n,c;
cin>>n>>c;
for(lon i=;i<=n;++i)cin>>arr[i];
for(lon i=;i<=n;++i)
{
cin>>nex[i];
arr[i]-=nex[i];
}
for(int i=n+;i<=*n;++i)arr[i]=arr[i-n];
int sum=,cnt=,bg=;
bool ok=;
for(int i=;i<=*n;++i)
{
if(sum+arr[i]+c<)
{
for(;bg<=i&&sum+arr[i]+c<;++bg)
{
if(bg<i)sum-=arr[bg];
}
if(bg<=i)sum+=arr[i];
}
else
{
//if(bg==0)bg=i;
sum+=arr[i];
if(i-bg+>=n)
{
ok=;
break;
}
}
}
// cout<<" "<<ok<<endl;
if(ok)cout<<bg<<endl;
else cout<<-<<endl;
}
return ;
}