POJ 3181 Dollar Dayz

其实这道题就是简单的完全背包问题。从低到高推断出每种硬币的所有面额的解即可。当然,离线的话会慢一点。可以将问题需要求出的子问题全部求出来。下一个问题出现的时候,如果之前已经求解过则不必求解,否则在之前的基础上继续求解。原本觉得没什么好写的,关键是同样的方法用STL中的vector代替数组会超时!搞得以后都不敢用STL了.....

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <vector>
#include <cstring>
#define MAXN 1000+5
#define MAXK 100+5
#define MAXL 200
using namespace std;

char ans[MAXK][MAXN][MAXL];
int vis[MAXK][MAXN];
int L[MAXK][MAXN];//答案的长度记录
void add(int,int);
void doit(int,int,int,int);

int main()
{
    memset(vis,0,sizeof(vis));
    for(int i=0;i<MAXN;i++) {ans[1][i][1]='1';vis[1][i]=1;L[1][i]=1;}//K=1
    for(int i=1;i<MAXK;i++) {ans[i][0][1]='1';vis[i][0]=1;L[i][0]=1;}//N=0

    int N,K;
    int LK=1,LN=0;//LK,LN记录上次求解到的位置
    while(cin>>N>>K){
        if(!vis[K][N]) {doit(LK,LN,K,N);LK=K,LN=N;}
        for(int i=L[K][N];i>0;i--)
            cout<<ans[K][N][i];
        cout<<endl;
    }
}

void doit(int LK,int LN,int K,int N)
{
    for(int i=LK+1;i<=K;i++){
        for(int j=1;j<i;j++) {
            if(vis[i][j]) continue;
            for(int k=1;k<=L[i-1][j];k++) ans[i][j][k]=ans[i-1][j][k];
            L[i][j]=L[i-1][j];
            vis[i][j]=1;
        }
        for(int j=i;j<=N;j++) {
            if(vis[i][j]) continue;
            add(i,j);
            vis[i][j]=1;
        }
    }
}

void add(int k,int n)//高精度加法运算
{
    int len;

    for(int i=1;i<=L[k-1][n];i++) ans[k][n][i]=ans[k-1][n][i];//先复制一个序列
    L[k][n]=L[k-1][n];

    len=max(L[k][n],L[k][n-k]);

    int sum,overflow=0;
    for(int i=1;i<=len;i++){
        if(L[k][n-k]<i){//根据两序列长度不同分类讨论
            sum=ans[k][n][i]-'0'+overflow;
        }
        else {
            if(L[k][n]<i) {ans[k][n][i]='0';L[k][n]++;}
            sum=ans[k][n][i]-'0'+ans[k][n-k][i]-'0'+overflow;
        }
        ans[k][n][i]=sum%10+'0';
        overflow=sum/10;
    }
    if(overflow){//最后还有进位
        len=++L[k][n];
        ans[k][n][len]=overflow+'0';
    }

    return ;
}
时间: 2024-10-04 10:42:49

POJ 3181 Dollar Dayz的相关文章

POJ 3181 Dollar Dayz(完全背包+简单高精度加法)

POJ 3181 Dollar Dayz(完全背包+简单高精度加法) http://poj.org/problem?id=3181 题意: 给你K种硬币,每种硬币分别是1美元,2美元-K美元且可以无限使用,问你用上面K种硬币构成n美元的话有多少种方法? 分析: 本题是一道明显的完全背包问题, 不过本题还可以换一种方法来看: 整数n由前K个自然数构造, 一共有多少种方法? (虽然本题要用到高精度加法, 但是很简单, 不要被吓到哦) 首先是DP部分: 令dp[i][j]==x 表示由前i种硬币构成j

POJ 3181 Dollar Dayz &amp;&amp; Uva 147 Dollars(完全背包)

首先是 Uva 147:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=83 细心看完这题后发现还是完全背包,只不过需要对浮点数处理一下.即把所有硬币的面值都乘以100,化为整数,对输入的数据也作同样的处理,然后就是套完全背包的模板了,在输出时还要用格式和精度来卡一卡你……一开始我没想到用printf可以的,于是百度了cout的输出格式控制,

POJ 3181 Dollar Dayz 【完全背包】

题意: 给出两个数,n,m,问m以内的整数有多少种组成n的方法完全背包+大数划分 思路: dp[i][j] := 用i种价格配出金额j的方案数. 那么dp[i][0] = 1,使用任何价格配出金额0的方案个数都是1(什么都不用). 递推关系式: 实际上是完全背包问题,只是状态转移方程形式有所不同,不过状态转移的方向是完全相同的. dp[i][j] = dp[i – 1][j] + dp[i – 1][j – i] + dp[i – 1][j – 2 * i] + - + dp[i – 1][0]

poj 3181 Dollar Dayz DP

题意:给你一个n,还有k,求问有多少种数字组合,能够使得数字之和为n,这些数字的范围是1到k. 如,给你n=4, k=2.那么 1+1+1+1=4, 1+1+2=4,2+2=4,四种组合. 思路:完全背包,可以设d[i][j]代表从i个数字相加和为j的组合数. 那么,可以考虑把这些组合数分为,有数字i和没有数字i,那么没有数字i的组合数就为d[i-1][j],有数字i的组合数就为d[i][j-i](可以在这些组合里面加上1个i). 所以,转移方程可以写成d[i][j] = d[i-1][j] +

POJ 3181 Dollar Dayz 01全然背包问题

01全然背包问题. 主要是求有多少种组合.二维dp做的人多了,这里使用一维dp就能够了. 一维的转换方程:dp[j] = dp[j-i] + dp[j];当中i代表重量,j代表当前背包容量. 意思就是dp[j-i] 代表j-i背包重量的时候最多的组合数,那么假设到了背包容量为j的时候,就是能够把第i个物品装进背包,那么就有dp[j-i]种装法, 假设没有i物品之前.那么容量为j的时候组合数是dp[j]. 那么当有i物品,且容量为j的时候,那么组合数就是dp[j-i] + dp[j]; 二维能够转

POJ 3181 Dollar Dayz 01完全背包问题

01完全背包问题. 主要是求有多少种组合.二维dp做的人多了,这里使用一维dp就可以了. 一维的转换方程:dp[j] = dp[j-i] + dp[j];其中i代表重量,j代表当前背包容量. 意思就是dp[j-i] 代表j-i背包重量的时候最多的组合数,那么如果到了背包容量为j的时候,就是可以把第i个物品装进背包,那么就有dp[j-i]种装法, 如果没有i物品之前,那么容量为j的时候组合数是dp[j]: 那么当有i物品,且容量为j的时候,那么组合数就是dp[j-i] + dp[j]; 二维可以转

POJ 3181 Dollar Dayz ( 完全背包 &amp;&amp; 大数高精度 )

题意 : 给出目标金额 N ,问你用面额 1~K 拼成 N 的方案有多少种 分析 : 完全背包的裸题,完全背包在 DP 的过程中实际就是列举不同的装填方案数来获取最值的 故状态转移方程为 dp[i] += dp[j-w[i]] 但是这题怎么可能那么简单呢! N 和 K 的上限导致答案过大,需要使用高精度加法来完成 所以无耻的用 JAVA 来搞定了 import java.io.*; import java.lang.reflect.Array; import java.util.*; impor

poj3181 Dollar Dayz (DP+大数)

Dollar Dayz Crawling in process... Crawling failed Time Limit:1000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u Submit Status Practice POJ 3181 Appoint description: System Crawler (2016-05-27) Description Farmer John goes to Dollar Da

POJ - Counterfeit Dollar 题解

挺考智力的题目. 思路: 1 如果是假币,那么每次都必定引起天平的不平衡 2 如果天平平横,那么全部都肯定是真币 利用这个特性,利用hash表,就能写出很简洁的程序. 如果使用枚举,那么会(轻松?)过百行的代码的. 当然其实题目给出了条件:一定可以找出唯一的假币的. 如果没有这个条件,那么是不一定可以三次称,就能确定结果的. 下面程序参考了别人的: http://www.cnblogs.com/orangeman/archive/2009/07/10/1520663.html 这个家伙的思路也不