Description
有一个长为n(n≤500)n(n≤500)n(n≤500)的序列,每次可以选一段区间加上xxx,如果有数大于777就减777(保持所有数∈[1,7]∈[1,7]∈[1,7])
问:最少几次这样的操作才能使所有aia_iai都等于777
Solution
step1:step1:step1:先差分一下,令bi=ai−ai−1(1≤i≤n+1,a0=an+1=0)b_i=a_i-a_{i-1}(1≤i≤n+1,a_0=a_{n+1}=0)bi=ai−ai−1(1≤i≤n+1,a0=an+1=0)
每次操作就变成bl+xb_l+xbl+x,br+1−xb_{r+1}-xbr+1−x,目的是让所有bi=0b_i=0bi=0
step2:step2:step2:我们发现111和666是可以一次抵消掉的,同理,222和555、333和444也一样
暴力消的话,也最多消数的个数次就行
step3:dpstep3:dpstep3:dp
因为111和666、222和555、333和444每对都有一个数被抵消掉了,所以dpdpdp状态只要四维就行(第四维是记录当前这些数的和)
设f[i][j][k][l]f[i][j][k][l]f[i][j][k][l]表示已经用了iii个1/61/61/6,jjj个2/52/52/5,kkk个3/43/43/4,总和mod7mod7mod7为lll时,最多能少几个步骤,转移见代码
复杂度O(7⋅(n6)3)O(7\cdot(\frac{n}{6})^3)O(7⋅(6n)3)
Code
#include<bits/stdc++.h>
using namespace std;
int n,x,y,z,i,mx,my,mz,l,j,k,a[8],f[2][502][502][7],las,ans,p;
void Max(int &x,int y){if(y>x)x=y;}
int main(){
scanf("%d",&n);
for (i=1;i<=n;i++){
scanf("%d",&x);
a[(x-las+7)%7]++,las=x;
}
a[7-las]++;
if (a[1]>a[6]) x=1,swap(a[1],a[6]);
else x=6;
if (a[2]>a[5]) y=2,swap(a[2],a[5]);
else y=5;
if (a[3]>a[4]) z=3,swap(a[3],a[4]);
else z=4;
ans=a[4]+a[5]+a[6];
mx=a[6]-a[1],my=a[5]-a[2],mz=a[4]-a[3];
for (i=0;i<=mx;i++,p^=1)
for (j=0;j<=my;j++)
for (k=0;k<=mz;k++)
for (l=0;l<7;l++)
Max(f[p^1][j][k][(l+x)%7],f[p][j][k][l]+(l+x==7)),
Max(f[p][j+1][k][(l+y)%7],f[p][j][k][l]+(l+y==7)),
Max(f[p][j][k+1][(l+z)%7],f[p][j][k][l]+(l+z==7));
printf("%d",ans-f[p^1][my][mz][0]);
}