区间dp(整数划分,石子划分)

整数划分(四)

链接: http://acm.nyist.net/JudgeOnline/problem.php?pid=746
基础区间dp,代码:
#include <cstdio>
#include <cstring>
long long a[20][20];
long long dp[25][25];
int main()
{
    int T,m;
    scanf("%d",&T);
    getchar();
    while(T--)
    {
        char s[22];
        scanf("%s",s+1);
        scanf("%d",&m);
        int l=strlen(s),ok=1;
        memset(a,0,sizeof(a));
        for(int i=1;i<l;i++)
        {
            if(s[i]=='0')
                ok=0;
            for(int j=i;j<l;j++)
            {
                a[i][j]=a[i][j-1]*10+(s[j]-'0');
            }
        }
        if(ok==0&&l-1==m||l-1<m)
        {
            printf("0\n");continue;
        }
        long long x,ans;
        memset(dp,0,sizeof(dp));
        for(int i=0;i<l;i++)
            dp[i][1]=a[1][i];
        ans=0;
        if(m==1)
            ans=dp[l-1][1];
        for(int j=2;j<=m;j++)
        {
            for(int i=j;i<l;i++)
            {
                ans=a[i][i];
                for(int k=1;k<i;k++)
                {
                    x=dp[k][j-1]*a[k+1][i];
                    if(x>ans)
                        ans=x;
                }
                dp[i][j]=ans;
            }
        }
        printf("%lld\n",ans);
    }
    return 0;
}

石子合并(一)

链接: http://acm.nyist.net/JudgeOnline/problem.php?pid=737
代码:
 
#include <cstdio>
#include <cstring>
#define N 210
int dp[N][N],sum[N];
int main()
{
    int n;
    while(~scanf("%d",&n))
    {
        int a[N];sum[0]=0;
        for(int i=1;i<=n;i++){
            scanf("%d",&a[i]);
            sum[i]=sum[i-1]+a[i];
        }
        memset(dp,0,sizeof(dp));
        int i,j,l,k;
		for(l = 2; l <= n; ++l)
		{
			for(i = 1; i <= n - l + 1; ++i)
			{
				j = i + l - 1;
				dp[i][j] = 2100000000;
				for(k = i; k < j; ++k)
				{
					int q = dp[i][k] + dp[k + 1][j] + sum[j] - sum[i-1];
					if(dp[i][j] > q)
						dp[i][j] = q;
				}
			}
		}
		printf("%d\n", dp[1][n]);
    }
    return 0;
}
        



你可能感兴趣的:(Algorithm,优化,动态规划,合并)