DAG模型——硬币问题
硬币问题
有n种硬币,面值分别为V1,V2,...,Vn,每种都有无限多。给定非负整数S,可以选用多少个硬币,使得面值之和恰好为S?输出硬币数目的最小值和最大值。1<=n<=100, 0<=S<=10000,1<=Vi<=S.
分析:
我们把每种面值看做一个点,表示“还需要凑足的面值”,则初始状态为S,目标状态为0.若当前在状态 i ,每使用一个硬币j ,状态便转移到 i-Vj .
注意到最长路和最短路的求法是类似的,下面只考虑最长路。由于终点固定,d(i)的确切含义变为“从结点i出发到结点0的最长路径长度”
在记忆化搜索中,如果用特殊值表示“还没算过”,则必须将其和其他特殊值(比如无解)区分开。
也可以不用特殊值表示还没算过,而是用另外一个数组vis[i]表示状态i “是否被访问过”
记忆化搜索代码如下:
#include<cstdio>
#include<cstdlib>
#include<algorithm>
using namespace std;
const int maxn = , maxv = ;
int n, S, v[maxn], mind[maxv], maxd[maxv];
int dp1(int s) //最小值
{
int &ans = mind[s];
if(ans != -) return ans;
ans = <<;
for(int i = ; i <= n; ++i) if(v[i] <= s) ans = min(ans, dp1(s-v[i])+);
return ans;
}
int vis[maxv];
int dp2(int s) //最大值
{
if(vis[s]) return maxd[s];
vis[s] = ;
int &ans = maxd[s];
ans = -<<;
for(int i = ; i <= n; ++i) if(s >= v[i]) ans = max(ans, dp2(s-v[i])+);
return ans;
}
void print_ans(int* d, int s) //打印的是边
{
for(int i = ; i <= n; ++i) if(v[i] <= s && d[s-v[i]]+ ==d[s])
{
printf("%d ",i);
print_ans(d, s-v[i]);
break;
}
}
int main()
{
freopen("9-3.in", "r", stdin);
scanf("%d%d", &n, &S);
for(int i = ; i <= n; ++i) scanf("%d", v+i);
memset(mind, -, sizeof(mind));
mind[] = ;
printf("%d\n", dp1(S));
print_ans(mind, S);
printf("\n");
memset(maxd, -, sizeof(maxd));
memset(vis, , sizeof(vis));
maxd[] = ;
vis[] = ;
printf("%d\n", dp2(S));
print_ans(maxd, S);
printf("\n");
return ;
}
递推代码如下:
#include <cstdio>
#include <cstring>
const int maxn = , maxv = , INF = ;
int v[maxn], S, n, maxf[maxv], minf[maxv], min_coin[maxv], max_coin[maxv];
void print_ans(int *d, int s){
while(s){
printf("%d ", d[s]);
s-=v[d[s]];
}
}
int main(){
freopen("9-3.in", "r", stdin);
scanf("%d%d", &n, &S);
for(int i = ; i <= n; ++i) scanf("%d", &v[i]);
for(int i = ; i <= S; ++i) {
//最应注意的是边界条件、初始化,因为递推作为重点一般不会写错,就是在这种细节处出错
maxf[i] = -INF;
minf[i] = INF;
}
maxf[] = minf[] = ;
for(int i = ; i <= S; ++i)
for(int j = ; j <= n; ++j) if(i >= v[j]){
if(maxf[i-v[j]] + > maxf[i]){
maxf[i] = maxf[i-v[j]] + ;
max_coin[i] = j;
}
if(minf[i-v[j]] + < minf[i]){
minf[i] = minf[i-v[j]] + ;
min_coin[i] = j;
}
}
printf("%d\n", minf[S]);
print_ans(min_coin, S);
printf("\n");
printf("%d\n", maxf[S]);
print_ans(max_coin, S);
printf("\n");
return ;
}
不必打印路径代码:
#include<cstdio>
#include<cstdlib>
const int maxn = , maxv = , INF = ;
int n, S, v[maxn], min[maxv], max[maxv];
int main(){
scanf("%d%d", &n, &S);
for(int i = ; i <= n; ++i) scanf("%d", v+i);
min[] = max[] = ;
for(int i = ; i <= S; ++i) {min[i] = INF; max[i] = -INF;}
for(int i = ; i <= S; ++i)
for(int j = ; j <= n; ++j) if(v[j] <= i){
if(min[i-v[j]] + < min[i]) min[i] = min[i-v[j]] + ;
if(max[i-v[j]] + > max[i]) max[i] = max[i-v[j]] + ;
}
printf("%d\n", min[S]);
printf("%d\n", max[S]);
return ;
}
DAG模型——硬币问题的更多相关文章
- UVA103 dp基础题,DAG模型
1.UVA103 嵌套n维空间 DAG模型记忆化搜索,或者 最长上升子序列. 2.dp[i]=max( dp[j]+1),(第i个小于第j个) (1) //DAG模型记忆化搜索 #include< ...
- NYOJ16|嵌套矩形|DP|DAG模型|记忆化搜索
矩形嵌套 时间限制:3000 ms | 内存限制:65535 KB 难度:4 描述 有n个矩形,每个矩形可以用a,b来描述,表示长和宽.矩形X(a,b)可以嵌套在矩形Y(c,d)中当且仅当a& ...
- DAG模型
数字三角形: 1.递归计算 int solve(int i,int j) { :max(solve(i+,j),solve(i+,j+))); } 2.记忆化搜索,不用指明计算顺序,并且保证每个状态只 ...
- DAG模型——嵌套矩阵
有向无环图上的动态规划是学习动态规划的基础,很多问题都可以转化为DAG上的最长路.最短路或路径计数问题. 嵌套矩阵 有n个矩阵,每个矩阵可以用两个整数a,b描述,表示它的长和宽.矩阵X(a,b)可以嵌 ...
- DAG动态规划-硬币问题
题目:有n种硬币,面值分别为V1,V2,...Vn,每种都有无限多.给定非负整数S,可以选用多少个硬币,使得面值之和恰好为S?输出硬币数目的最小值和最大值! #include <bits/std ...
- DAG 模型 stacking boxes 动态规划
题目:UVA 103 stacking boxes 题目大意: 给你两个数,一个是盒子的个数,一个是每一个盒子的维数.将一个个盒子互相装起来,让你求最多可以装多少个,要求字典序最小. 解析:这个就是盒 ...
- DAG模型(矩形嵌套)
推荐在线例题:http://acm.nyist.net/JudgeOnline/problem.php?pid=16 题摘: 矩形嵌套 时间限制:3000 ms | 内存限制:65535 KB 难 ...
- DP入门(2)——DAG上的动态规划
有向无环图(DAG,Directed Acyclic Graph)上的动态规划是学习动态规划的基础.很多问题都可以转化为DAG上的最长路.最短路或路径计数问题. 一.DAG模型 [嵌套矩形问题] 问题 ...
- 第九章(二)DAG上的动态规划
DAG上的动态规划: 有向无环图上的动态规划是学习DP的基础,很多问题都可以转化为DAG上的最长路.最短路或路径计数问题. 1.没有明确固定起点重点的DAG模型: 嵌套矩形问题:有n个矩形,每个矩形可 ...
随机推荐
- weka 集成学习
import java.io.*;import weka.classifiers.*;import weka.classifiers.meta.Vote;import weka.core.Instan ...
- 冒泡算法(C++模板实现)
冒泡排序 从整体上看,冒泡排序是一种稳定排序,即排序完成后,原本序列中的键值相等的元素相对位置不会发生改变.算法的时间复杂度是O(n2),空间复杂度为O(1),即这是一个"就地算法" ...
- 归并排序,递归法,C语言实现。
利用归并排序法对序列排序的示意图(递归法): 一.算法分析:利用递归的分治方法:1.将原序列细分,直到成为单个元素:2.在将分割后的序列一层一层地按顺序合并,完成排序.细分通过不断深入递归完成,合并通 ...
- 细说Oracle数据库与操作系统存储管理二三事
在上大学的时候,学习操作系统感觉特别枯燥,都是些条条框框的知识点,感觉和实际应用的关联不大.发现越是工作以后,在工作中越想深入了解,发现操作系统知识越发重要.在实践中结合理论还是不错的一种学习方法.自 ...
- PC-IIS因为端口问题报错的解决方法
1.我的电脑-管理-服务和应用程序-Internet信息服务 情况:这时发现“默认 SMTP 虚拟服务器”停止 解决方法:右击启动 情况:发现网页还是打不开.2.Internet信息服务-网站- ...
- go语言与所谓的包
import后面接的是目录的名字,而不是所谓包的名字,并且如果一个目录下面还有目录的话都必须要写进去,比如: import "MyPackage" import "MyP ...
- poj3101
不难,结果: 程序: import java.math.*; import java.util.*; public class Main { public static void main(Strin ...
- 如何在Windows中打开多个Windows Media Player
博客搬到了fresky.github.io - Dawei XU,请各位看官挪步.最新的一篇是:如何在Windows中打开多个Windows Media Player.
- android_ViewPager_实现导航页
android_ViewPager_实现导航页 既然是实现导航页的效果,那么我们肯定是要实现ViewPager的 要实现的效果如下 1.用户进入欢迎页面 2.判断是否是第一次进入,如果是,则进入导航页 ...
- cocos2d-x游戏开发实战原创视频讲座系列1之2048游戏开发
cocos2d-x游戏开发实战原创视频讲座系列1之2048游戏开发 的产生 视持续更新中.... 视频存放地址例如以下:http://ipd.pps.tv/user/1058663622 ...