Luogu P1043 数字游戏
经过第二天平复心情后回来写题解…
写bug5分钟 debug2小时
显然,区间dp,开2倍长度+前缀和
f[i][j][t]表示区间(i,j)分成t段所得到的最大值。
区间长度、起点、分段数、断点,四重循环。
注意范围——比如分段数的上限是min(m,j-i+1).
*之前的枚举顺序不对但是过了,我也不知道为什么…
#include<cstdio> #include<iostream> #include<cmath> #include<cstring> #define MogeKo qwq using namespace std; const int maxn = 105; const int INF = 0x3f3f3f3f; int n,m; long long ans1,ans2,a[maxn],f[maxn][maxn][10],g[maxn][maxn][10]; int main() { scanf("%d%d",&n,&m); for(int i = 1; i <= n; i++) { scanf("%lld",&a[i]); a[i+n] = a[i]; } for(int t = 1; t <= m; t++) for(int i = 1; i <= n; i++) for(int j = 1; j <= i+n-1; j++) g[i][j][t] = INF; for(int i = 1; i <= n+n; i++) a[i] += a[i-1]; for(int i = 1; i <= n; i++) for(int j = i; j <= i+n-1; j++) f[i][j][1] = g[i][j][1] = ((a[j]-a[i-1])%10+10)%10; for(int len = 2; len <= n; len++) for(int i = 1; i <= n; i++) { int j = i+len-1; for(int t = 2; t <= min(m,j-i+1); t++) for(int k = i; k <= j-1; k++) { f[i][j][t] = max(f[i][j][t],f[i][k][t-1] * (((a[j]-a[k])%10+10)%10)); g[i][j][t] = min(g[i][j][t],g[i][k][t-1] * (((a[j]-a[k])%10+10)%10)); } } ans2 = INF; for(int i = 1; i <= n; i++) { ans1 = max(ans1,f[i][i+n-1][m]); ans2 = min(ans2,g[i][i+n-1][m]); } printf("%lld\n%lld\n",ans2,ans1); return 0; }