UVA 12105 Bigger is Better(数位dp)

f[i][j]表示用i根棍子余数为j可以组成的最大数。

但是最长可能有55位,所以用字符串来保存这个数,所以定义f[i][j][k];

每次先枚举火柴棍的根数,再枚举余数,最后枚举最新加进去的数字,每次把火柴棍为i时,余数为0的最大组成数字记下来即可。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
char f[105][3005][55];
#define rep(i,a,b) for(int i=(a);i<=(b);i++)
#define ss(x) scanf("%d",&x)
int n,m;
int need[10]={6,2,5,5,4,5,6,3,7,6};
int cmp(char *a,char *b)
{
    int la=(int)strlen(a);
    int lb=(int)strlen(b);
    if(la!=lb) return la<lb;
    return strcmp(a,b)<0;
}
void work(char *a,char *b,int k)
{
    char temp[55];
    memset(temp,0,sizeof(temp));
    strcpy(temp,b);
    if(temp[0]=='0') temp[0]=temp[0]+k;
    else temp[strlen(temp)]=k+'0';
    if(cmp(a,temp)) strcpy(a,temp);
}
int main()
{
    int kase=0;
    while(true)
    {
        ss(n);if(n==0) break;ss(m);
        memset(f,0,sizeof(f));
        f[0][0][0]='0';
        char ans[55];
        memset(ans,0,sizeof(ans));
        rep(i,0,n){
            rep(j,0,m){
                if(strlen(f[i][j])==0) continue;
                rep(k,0,9){
                    if(i+need[k]>n) continue;
                    int t=(j*10+k)%m;
                    work(f[i+need[k]][t],f[i][j],k);
                }
            }
            if(i&&cmp(ans,f[i][0])) memcpy(ans,f[i][0],sizeof(f[i][0]));
        }
        if(ans[0]=='\0') printf("Case %d: -1\n",++kase);
        else printf("Case %d: %s\n",++kase,ans);
    }
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值