分析:
環形要double轉換爲鏈狀,再跑dp
記dp[i]表示以 i 爲結尾的長度不超過k的最大連續子段和
即:
用單調隊列優化,O(1)轉移。
#include<cstdio>
#include<cstring>
#define MAXN 100000
#define INF 2000000000
struct node{
int val,pos;
}que[MAXN*2+10];
int n,a[MAXN*2+10],sum[MAXN*2+10],dp[MAXN*2+10],L[MAXN*2+10],k;
void Init()
{
memset(sum,0,sizeof sum);
memset(dp,0,sizeof dp);
memset(L,0,sizeof L);
memset(que,0,sizeof que);
}
void read()
{
scanf("%d%d",&n,&k);
for(int i=1;i<=n;i++){
scanf("%d",&a[i]);
a[i+n]=a[i];
}
n=2*n-1;
for(int i=1;i<=n;i++)
sum[i]=sum[i-1]+a[i];
}
void DP()
{
int front=0,rear=1; //que[0].val=0
for(int i=1;i<=n;i++){
while(front<rear){
if(que[front].pos<i-k)
front++;
else
break;
}
dp[i]=sum[i]-que[front].val;
L[i]=que[front].pos+1;
while(front<rear){
if(que[rear-1].val>=sum[i])
rear--;
else
break;
}
que[rear].pos=i,que[rear].val=sum[i];
rear++;
}
int ans1=-INF,ans2=-1,ans3=-1,ans4=-1;
int m=(n+1)/2;
for(int i=1;i<=n;i++){
if(dp[i]>ans1){
ans1=dp[i];
ans2=(L[i]%m)==0?m:(L[i]%m);
ans3=(i%m)==0?m:(i%m);
ans4=i-L[i];
}
else if(dp[i]==ans1){
int t=(L[i]%m==0)?m:(L[i]%m);
if(t<ans2){
ans2=t;
ans3=(i%m==0)?m:(i%m);
ans4=i-L[i];
}
else if(t==ans2){
if(i-L[i]<ans4){
ans3=(i%m)==0?m:(i%m);
ans4=i-L[i];
}
}
}
}
printf("%d %d %d\n",ans1,ans2,ans3);
}
int main()
{
int T;
scanf("%d",&T);
while(T--){
Init();
read();
DP();
}
}