题目描述
丁丁最近沉迷于一个数字游戏之中。这个游戏看似简单,但丁丁在研究了许多天之后却发觉原来在简单的规则下想要赢得这个游戏并不那么容易。游戏是这样的,在你面前有一圈整数(一共n个),你要按顺序将其分为m个部分,各部分内的数字相加,相加所得的m个结果对10取模后再相乘,最终得到一个数k。游戏的要求是使你所得的k最大或者最小。
例如,对于下面这圈数字(n=4,m=2):
要求最小值时,((2-1) mod 10)×((4+3) mod 10)=1×7=7,要求最大值时,为((2+4+3) mod 10)×(-1 mod 10)=9×9=81。特别值得注意的是,无论是负数还是正数,对10取模的结果均为非负值。
丁丁请你编写程序帮他赢得这个游戏。
输入输出格式
输入格式:
输入文件第一行有两个整数,n(1≤n≤50)和m(1≤m≤9)。以下n行每行有个整数,其绝对值不大于104,按顺序给出圈中的数字,首尾相接。
输出格式:
输出文件有两行,各包含一个非负整数。第一行是你程序得到的最小值,第二行是最大值。
输入输出样例
输入样例#1:
4 2
4
3
-1
2
输出样例#1:
7
81
dp方程表示从第i个到第j个数中取k段的最大值或最小值。状态转移方程
具体看代码
分段点,枚举两段分别所取的段数,取最值。
#include<cstdio>
#include<iostream>
#include<algorithm>
using namespace std;
#define INF 0x7fffffff
inline int mod(int x) {
return (x%10+10)%10;
}
int n,m;
int a[1000],sum[1000];
int dp[300][300][99][2];
int st[300][300];
int main() {
scanf("%d%d",&n,&m);
for(int i=1;i<=n;++i) {
scanf("%d",a+i);a[n+i]=a[i];
}
n<<=1;
for(int i=1;i<=n;++i) {
sum[i]=a[i]+sum[i-1];
}
for(int i=0;i<130;++i)
for(int j=0;j<130;++j)
for(int k=0;k<130;++k) {
dp[i][j][k][1]=INF;
}
for(int i=1;i<=n;i++)
for(int j=i;j<=n;j++) {
dp[i][j][1][1]=dp[i][j][1][0]=st[i][j]=((sum[j]-sum[i-1])%10+10)%10;
}
for(int t=2;t<=m;++t) {
for(int i=1;i<=n;++i) {
for(int j=i;j<=n;++j) {
for(int k=i;k<j;++k) {
dp[i][j][t][0]=max(dp[i][j][t][0],dp[i][k][t-1][0]*st[k+1][j]);
if(dp[i][k][t-1][1]<INF)dp[i][j][t][1]=min(dp[i][j][t][1],dp[i][k][t-1][1]*st[k+1][j]);
}
}
}
}
n>>=1;
int maxans=-10000,minans=0x7fffffff;
for(int i=1;i<=n;++i)
{
maxans=max(maxans,dp[i][i+n-1][m][0]);
//printf("%d
",dp[i][i+n-1][m][1]);
minans=min(minans,dp[i][i+n-1][m][1]);
}
printf("%d
%d
",minans,maxans);
return 0;
}
/*
4 2
4
3
-1
2
*/