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美元一共有x种方法.

初始化dp全0 且 dp[0][0]=1

状态转移: dp[i][j] = sum( dp[i-1][j] , dp[i][j-val[i]] )    //sum为求和,val[i]是第i种硬币的面值.

前者表示第i种硬币一个都不选, 后者表示至少选一个第i种硬币来用.

最终所求: dp[k][n]的值. 程序实现用的滚动数组, 所以dp只有[j]一维.

其次是高精度部分:

如果输入1000 100时,输出将为:

15658181104580771094597751280645这个值超过了long long的范围. 所以这里我们需要用大整数来表示dp[i][j]的值. 我的大整数实现是用high和low 的组合来表示一个大整数的. 其中low表示大整数的十进制表示时的低18位数. high表示大整数的十进制表示时的高18位数.

总体来说实现比较简单.

AC代码:

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const long long BASE = 1e18;
const int maxn=1000+5;

int n,k;

//大整数类
class BigNum
{
public:
    BigNum(){}
    BigNum(long long high,long long low):high(high),low(low){}
    long long high; //高18位
    long long low;  //低18位

    //相加运算
    BigNum operator+(BigNum &B)
    {
        long long high_tmp = (low+B.low)/BASE+high+B.high;
        long long low_tmp = (low+B.low)%BASE;
        return BigNum(high_tmp, low_tmp);
    }

    //输出值
    void print()
    {
        if(!high)//高位为0
            printf("%I64d\n",low);
        else     //高位非0
        {
            printf("%I64d",high);
            printf("%018I64d",low);
        }
    }
}dp[maxn];

int main()
{
    while(scanf("%d%d",&n,&k)==2)
    {
        //初始化
        memset(dp,0,sizeof(dp));
        dp[0].low=1;//等效于令dp[0]=0;

        //递推
        for(int i=1;i<=k;i++)
        {
            for(int j=i;j<=n;j++)
                dp[j] = dp[j]+dp[j-i];
        }

        //输出
        dp[n].print();
    }

    return 0;
}
时间: 2024-12-13 06:25:44

POJ 3181 Dollar Dayz(完全背包+简单高精度加法)的相关文章

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

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

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

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]; 二维可以转

问题 B: 【高精度】简单高精度加法

问题 B: [高精度]简单高精度加法 时间限制: 1 Sec  内存限制: 64 MB提交: 94  解决: 27[提交][状态][讨论版] 题目描述 修罗王解决了计算机的内存限制问题,终于可以使用电脑进行大型的魔法运算了,他交给邪狼的第一个任务是计算两个非负整数A.B的和,其中A和B的位数在5000位以内. 输入 共两行数据,第一行为一个非负整数A,第二行为一个非负整数B,A.B的位数均在5000以内. 输出 输出一个非负数,即两数之和. 样例输入 1111111111 2222222222

中石油-【高精度】简单高精度加法

问题 B: [高精度]简单高精度加法 时间限制: 1 Sec  内存限制: 64 MB提交: 63  解决: 20[提交][状态][讨论版] 题目描述 修罗王解决了计算机的内存限制问题,终于可以使用电脑进行大型的魔法运算了,他交给邪狼的第一个任务是计算两个非负整数A.B的和,其中A和B的位数在5000位以内. 输入 共两行数据,第一行为一个非负整数A,第二行为一个非负整数B,A.B的位数均在5000以内. 输出 输出一个非负数,即两数之和. 样例输入 1111111111 2222222222