code1047 邮票面值设计

dfs+dp

dfs枚举每种情况,每层递归确定第k个数i:i = a[k-1]+1 to a[k-1]*n+1

当枚举完一个序列时,使用check()测试它能达到的max

使用dp。设dp[i]为凑成面值为i的最少张数

for(int k=1;a[k]<=i&&k<=t;k++){
    if(dp[i-a[k]]<n){
        dp[i]=min(dp[i], dp[i-a[k]]+1);
    }
}

注意dp[i]一开始要设成最大值

代码如下:

#include<iostream>
#include<cstring>
#define Size 41
using namespace std;

int a[Size];
int dp[10005];
int best[Size];
int n,t;//n张 t种
int num=0;void save(int x){
    num=x;
    for(int i=1;i<=t;i++){
        best[i]=a[i];
    }
}

int check(){
    //memset(dp,0x3f,sizeof(dp));
    dp[0]=0;
    int ans=0;
    bool ok=true;

    //for(int i=1;i<=t;i++)cout<<a[i]<<" ";
    //cout<<endl;

    for(int i=1;ok;i++){
        ok=false;
        dp[i]=0x3f3f3f3f;
        for(int k=1;a[k]<=i&&k<=t;k++){
            if(dp[i-a[k]]<n){
                dp[i]=min(dp[i], dp[i-a[k]]+1);
                ok=true;
            }
        }
        if(ok)ans=i;

        //cout<<dp[i]<<‘ ‘;
    }
    //cout<<endl<<endl;

    return ans;
}

void dfs(int k){if(k>t){
        int x=check();
        if(x>num)save(x);
        return;
    }

    for(int i=a[k-1]+1;i<=n*a[k-1]+1;i++){
        a[k]=i;
        dfs(k+1);
    }
}

int main(){
    cin>>n>>t;

    a[1]=1;
    num=1;

    dfs(2);

    for(int i=1;i<=t;i++){
        cout<<best[i]<<‘ ‘;
    }
    cout<<endl<<"MAX="<<num<<endl;
    cout<<f;
    return 0;
} 

By the way:a[1]一定是1!

PS:你可以看到我把check的第一行memset注释掉了,因为大部分时候很长的dp数组后面的很多元素都没有用(max没有那么大),所以只需要在循环里用到一个写一个dp[i]=x03f3f3f3f就可以了。memset使程序变得很慢很慢。(真的很慢啊...)(你真的不需要再试它有多慢了,因为我已经花一个小时帮你试了无数遍了...)

如果想继续优化,可以边dfs边维护dp和当前max2的值,这样dfs循环就可以写成这样:

i = a[k-1]+1 to max2+1

如果i>max2+1,那么max2+1这个数将始终无法凑成,结果不会更优

时间: 2024-10-12 11:37:49

code1047 邮票面值设计的相关文章

[NOIP1999提高] CODEVS 1047 邮票面值设计(dfs+dp)

dfs出邮票的各种面值,然后dp求解. ------------------------------------------------------------------------------- #include<cstdio> #include<cstring> #include<algorithm> #include<vector> #define rep(i,n) for(int i=0;i<n;++i) #define clr(x,c) m

P1021 邮票面值设计

P1021 邮票面值设计 题目描述 给定一个信封,最多只允许粘贴N张邮票,计算在给定K(N+K≤15)种邮票的情况下(假定所有的邮票数量都足够),如何设计邮票的面值,能得到最大值MAX,使在1-MAX之间的每一个邮资值都能得到. 例如,N=3,K=2,如果面值分别为1分.4分,则在1分-6分之间的每一个邮资值都能得到(当然还有8分.9分和12分):如果面值分别为1分.3分,则在1分-7分之间的每一个邮资值都能得到.可以验证当N=3,K=2时,7分就是可以得到的连续的邮资最大值,所以MAX=7,面

深搜+DP剪枝 codevs 1047 邮票面值设计

codevs 1047 邮票面值设计 1999年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description 给定一个信封,最多只允许粘贴N张邮票,计算在给定K(N+K≤40)种邮票的情况下(假定所有的邮票数量都足够),如何设计邮票的面值,能得到最大值MAX,使在1-MAX之间的每一个邮资值都能得到. 例如,N=3,K=2,如果面值分别为1分.4分,则在1分-6分之间的每一个邮资值都能得到(当然还有8分.9分和1

codves1047:邮票面值设计

1047 邮票面值设计 1999年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 题目描述 Description 给定一个信封,最多只允许粘贴N张邮票,计算在给定K(N+K≤40)种邮票的情况下(假定所有的邮票数量都足够),如何设计邮票的面值,能得到最大值MAX,使在1-MAX之间的每一个邮资值都能得到. 例如,N=3,K=2,如果面值分别为1分.4分,则在1分-6分之间的每一个邮资值都能得到(当然还有8分.9分和12分):

邮票面值设计NOIP1999T

邮票面值设计NOIP1999T 问题描述给定一个信封,最多只允许粘贴N张邮票,计算在给定K(N+K≤40)种邮票的情况下(假定所有的邮票数量都足够),如何设计邮票的面值,能得到最大值MAX,使在1-MAX之间的每一个邮资值都能得到. 例如,N=3,K=2,如果面值分别为1分.4分,则在1分-6分之间的每一个邮资值都能得到(当然还有8分.9分和12分):如果面值分别为1分.3分,则在1分-7分之间的每一个邮资值都能得到.可以验证当N=3,K=2时,7分就是可以得到的连续的邮资最大值,所以MAX=7

CODEVS1047 邮票面值设计

题目描述 Description 给定一个信封,最多只允许粘贴N张邮票,计算在给定K(N+K≤40)种邮票的情况下(假定所有的邮票数量都足够),如何设计邮票的面值,能得到最大值MAX,使在1-MAX之间的每一个邮资值都能得到. 例如,N=3,K=2,如果面值分别为1分.4分,则在1分-6分之间的每一个邮资值都能得到(当然还有8分.9分和12分):如果面值分别为1分.3分,则在1分-7分之间的每一个邮资值都能得到.可以验证当N=3,K=2时,7分就是可以得到的连续的邮资最大值,所以MAX=7,面值

邮票面值设计(codevs 1047) 题解

[问题描述] 给定一个信封,最多只允许粘贴N张邮票,计算在给定K(N+K≤40)种邮票的情况下(假定所有的邮票数量都足够),如何设计邮票的面值,能得到最大值MAX,使在1-MAX之间的每一个邮资值都能得到.例如,N=3,K=2,如果面值分别为1分.4分,则在1分-6分之间的每一个邮资值都能得到(当然还有8分.9分和12分):如果面值分别为1分.3分,则在1分-7分之间的每一个邮资值都能得到.可以验证当N=3,K=2时,7分就是可以得到的连续的邮资最大值,所以MAX=7,面值分别为1分.3分. [

1047 邮票面值设计 (DFS+DP)

题目描述 Description 给定一个信封,最多只允许粘贴N张邮票,计算在给定K(N+K≤40)种邮票的情况下(假定所有的邮票数量都足够),如何设计邮票的面值,能得到最大值MAX,使在1-MAX之间的每一个邮资值都能得到. 例如,N=3,K=2,如果面值分别为1分.4分,则在1分-6分之间的每一个邮资值都能得到(当然还有8分.9分和12分):如果面值分别为1分.3分,则在1分-7分之间的每一个邮资值都能得到.可以验证当N=3,K=2时,7分就是可以得到的连续的邮资最大值,所以MAX=7,面值

邮票面值设计

题目: Description给定一个信封,最多只能充许粘贴N张邮票,计算在给定K(N+K<=40)种邮票的情况下(假定所有的邮票都足够),如何设计邮票的面值,能得到最大值max,使在1-max之间的每一个邮资值都能得到.例如,N=3,K=2,如果面值分别为1分,4分,则在1分~6分之间的每一个邮资都能得到(当然还有8分.9分和12分):如果面值分别为1分.3分,则在1分~7分之间的每一个邮资值都能得到.可以验证当N=3,K=2时,7分就是可以得到的连续的邮资最大值,所以max=7,面值分别为1