CJOJ 1071 【Uva】硬币问题(动态规划)
CJOJ 1071 【Uva】硬币问题(动态规划)
Description
有n种硬币,面值分别为v1, v2, ..., vn,每种都有无限多。给定非负整数S,可以选用多少个硬币,使得面值之和恰好为S?输出硬币数目的最小值和最大值。
Input
第一行两个整数,n,S(1≤n≤100, 0≤S≤100000)。
第二行n个整数vi-1...n(1≤vi≤S)。
Output
第一行两个整数,分别表示硬币数目的最小值 a 和最大值 b 。无解则输出 -1 。
第二行 a 个整数分别表示使用的是第几种硬币。
第三行 b 个整数分别表示使用的是第几种硬币。
Sample Input
6 12
1 2 3 4 5 6
Sample Output
2 12
6 6
1 1 1 1 1 1 1 1 1 1 1 1
Http
CJOJ:http://oj.changjun.com.cn/problem/detail/pid/1071
Source
动态规划
解决思路
看到这道题首先想到的是spfa算法。令每一个面值为一个点,若面值j=i+v[k]则连一条边(只是这么想,实际不用连边),那么分别用spfa跑出0->S的最短路径和最长路径。
这么想虽然没错,但会超时(70分)。下面会给出spfa代码,这里就不过多叙述。
那么正解是什么呢?动态规划。
我们以求最小值为例,定义Dmin[i]表示能凑出面值i的最少硬币数,用path_min[i]记录这个数量是从哪个面值转移过来的。那么我们就有Dmin[i]=min(Dmin[i-V[j]]+1)。
最后求路径的时候先令一个变量k=S,每次将k-path_min[k]压入一个vector中,再将k=path_min[k],直到k==0。然后将vector排序,再输出。
而最大值的操作是类似的。
另外要注意的是,题中输出的应该是面值的编号,所以用一个Map[i]存下面值i对应的编号,所以压入vector的就是Map[k-path_min[k]]
代码
spfa(TLE代码)
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<queue>
#include<vector>
using namespace std;
const int maxN=200;
const int maxS=100050;
const int inf=2147483647;
int n,S;
int V[maxN];
int Dmax[maxS];
int Dmin[maxS];
int path_max[maxS];//记路径
int path_min[maxS];
bool vis[maxS];
int Map[maxS];//因为最后要输出的是编号,所以为每一个面值都存一个编号
queue<int> Q;
vector<int> arr;
void spfa_min();
void spfa_max();
void dfs_min(int sum);//dfs求答案,但会爆栈
void dfs_max(int sum);
int main()
{
memset(Dmax,-1,sizeof(Dmax));
memset(Dmin,-1,sizeof(Dmin));
memset(Map,-1,sizeof(Map));
cin>>n>>S;
for (int i=1;i<=n;i++)
{
cin>>V[i];
if (Map[V[i]]==-1)
Map[V[i]]=i;//记录面值i所对应的的编号i
}
spfa_min();
spfa_max();
cout<<Dmin[S]<<' '<<Dmax[S]<<endl;
//dfs_min(S);cout<<endl;
//dfs_max(S);
int k=S;//手动写栈,排序
while (k>0)
{
arr.push_back(Map[k-path_min[k]]);
k=path_min[k];
}
sort(arr.begin(),arr.end());
for (int i=0;i<arr.size();i++)
cout<<arr[i]<<' ';
cout<<endl;
k=S;//手动写栈,排序
arr.clear();
while (k>0)
{
//cout<<":AA"<<endl;
arr.push_back(Map[k-path_max[k]]);
k=path_max[k];
}
sort(arr.begin(),arr.end());
for (int i=0;i<arr.size();i++)
cout<<arr[i]<<' ';
cout<<endl;
return 0;
}
void spfa_min()
{
memset(vis,0,sizeof(vis));
Q.push(0);
vis[0]=1;
Dmin[0]=0;
do
{
int u=Q.front();
Q.pop();
vis[u]=0;
for (int i=1;i<=n;i++)
if (u+V[i]<=S)
if ((Dmin[u]+1<Dmin[u+V[i]])||(Dmin[u+V[i]]==-1))
{
Dmin[u+V[i]]=Dmin[u]+1;
path_min[u+V[i]]=u;
if (vis[u+V[i]]==0)
{
Q.push(u+V[i]);
vis[u+V[i]]=1;
}
}
}
while (!Q.empty());
return;
}
void spfa_max()
{
memset(vis,0,sizeof(vis));
while (!Q.empty())
Q.pop();
Q.push(0);
vis[0]=1;
Dmax[0]=0;
do
{
int u=Q.front();
Q.pop();
vis[u]=0;
for (int i=1;i<=n;i++)
if (u+V[i]<=S)
if ((Dmax[u]+1>Dmax[u+V[i]])||(Dmax[u+V[i]]==-1))
{
Dmax[u+V[i]]=Dmax[u]+1;
path_max[u+V[i]]=u;
if (vis[u+V[i]]==0)
{
vis[u+V[i]]=1;
Q.push(u+V[i]);
}
}
}
while (!Q.empty());
return;
}
void dfs_min(int sum)
{
if (sum==0)
return;
dfs_min(path_min[sum]);
cout<<Map[sum-path_min[sum]]<<' ';
return;
}
void dfs_max(int sum)
{
if (sum==0)
return;
dfs_max(path_max[sum]);
cout<<Map[sum-path_max[sum]]<<' ';
return;
}
动态规划
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
const int maxN=200;
const int maxS=200000;
const int inf=1000000000;
int n,S;
int V[maxN];
int Dmin[maxS];
int Dmax[maxS];
int path_min[maxS];
int path_max[maxS];
int Map[maxS];
vector<int> arr;//输出时要临时存放面值的编号并排序
int main()
{
memset(Map,-1,sizeof(Map));
cin>>n>>S;
for (int i=1;i<=n;i++)
{
cin>>V[i];
if (Map[V[i]]==-1)
Map[V[i]]=i;
}
for (int i=1;i<=S;i++)
{
Dmin[i]=inf;//初始值,最小值最大,最大值最小
Dmax[i]=-inf;
}
Dmin[0]=0;
Dmax[0]=0;
for (int i=1;i<=S;i++)
{
for (int j=1;j<=n;j++)
if (i-V[j]<0)
continue;
else//DP
{
if (Dmin[i]>Dmin[i-V[j]]+1)
{
Dmin[i]=Dmin[i-V[j]]+1;
path_min[i]=i-V[j];
}
if (Dmax[i]<Dmax[i-V[j]]+1)
{
Dmax[i]=Dmax[i-V[j]]+1;
path_max[i]=i-V[j];
}
}
}
cout<<Dmin[S]<<' '<<Dmax[S]<<endl;
int k=S;
arr.clear();
while (k>0)//输出方案
{
arr.push_back(Map[k-path_min[k]]);
k=path_min[k];
}
sort(arr.begin(),arr.end());
for (int i=0;i<arr.size();i++)
cout<<arr[i]<<' ';
cout<<endl;
k=S;
arr.clear();
while (k>0)
{
arr.push_back(Map[k-path_max[k]]);
k=path_max[k];
}
sort(arr.begin(),arr.end());
for (int i=0;i<arr.size();i++)
cout<<arr[i]<<' ';
cout<<endl;
return 0;
}
CJOJ 1071 【Uva】硬币问题(动态规划)的更多相关文章
- CJOJ 2485 UVa 11991 生日礼物 / UVa 11991 Easy Problem from Rujia Liu?
CJOJ 2485 UVa 11991 生日礼物 / UVa 11991 Easy Problem from Rujia Liu? Description (原题来自刘汝佳<训练指南>Pa ...
- bzoj 2017 [Usaco2009 Nov]硬币游戏 动态规划
[Usaco2009 Nov]硬币游戏 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 431 Solved: 240[Submit][Status] ...
- [bzoj1708][Usaco2007 Oct]Money奶牛的硬币_动态规划_背包dp
Money奶牛的硬币 bzoj-1708 Usaco-2007 Oct 题目大意:在创立了她们自己的政权之后,奶牛们决定推广新的货币系统.在强烈的叛逆心理的驱使下,她们准备使用奇怪的面值.在传统的货币 ...
- CJOJ 1070 【Uva】嵌套矩形(动态规划 图论)
CJOJ 1070 [Uva]嵌套矩形(动态规划 图论) Description 有 n 个矩形,每个矩形可以用两个整数 a, b 描述,表示它的长和宽.矩形 X(a, b) 可以嵌套在矩形 Y(c, ...
- UVa 104 - Arbitrage(Floyd动态规划)
题目来源:https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=3&pa ...
- hdu 1284 分硬币 && uva 147
#include<bits/stdc++.h> using namespace std; int main() { unsigned ]; memset(dp,,sizeof(dp)); ...
- UVa 103 Stacking Boxes --- DAG上的动态规划
UVa 103 题目大意:给定n个箱子,每个箱子有m个维度, 一个箱子可以嵌套在另一个箱子中当且仅当该箱子的所有的维度大小全部小于另一个箱子的相应维度, (注意箱子可以旋转,即箱子维度可以互换),求最 ...
- uva 11324 The Largest Clique(强连通分量缩点+DAG动态规划)
http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=25&page=sh ...
- Bzoj 1042: [HAOI2008]硬币购物 容斥原理,动态规划,背包dp
1042: [HAOI2008]硬币购物 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1747 Solved: 1015[Submit][Stat ...
随机推荐
- 从LINQ开始之LINQ to Objects(下)
前言 上一篇<从LINQ开始之LINQ to Objects(上)>主要介绍了LINQ的体系结构.基本语法以及LINQ to Objects中标准查询操作符的使用方法. 本篇则主要讨论LI ...
- 小程序地图map
wxml: <button class="button" bindtap="getlocation" style="margin-top:30p ...
- 关于python编译的一点小结
大家都知道python是脚本语言,源码可以直接执行,有时需要提高执行效率或者保密(因为有时候不想让使用人看到源码文件),那就涉及到python编译了,那么该如何做呢? 有两种方法可以做到. 1.一种是 ...
- nodeJS之TCP模块net
前面的话 TCP服务在网络应用中十分常见,目前大多数的应用都是基于TCP搭建而成的.net模块提供了一个异步网络包装器,用于TCP网络编程,它包含了创建服务器和客户端的方法.本文将详细介绍nodeJS ...
- 【MyBatis源码解析】MyBatis一二级缓存
MyBatis缓存 我们知道,频繁的数据库操作是非常耗费性能的(主要是因为对于DB而言,数据是持久化在磁盘中的,因此查询操作需要通过IO,IO操作速度相比内存操作速度慢了好几个量级),尤其是对于一些相 ...
- html5shiv.min.js 和 respond.min.js 作用(bootstrap做IE低版本兼容时需要用到这两个插件)
1. html5shiv.min.js解决ie9以下浏览器对html5新增标签的不识别,并导致CSS不起作用的问题. respond.min.js让不支持css3 Media Query的浏览器包括I ...
- Linux centOS的vm虚拟机配置详细 中文版
这里以安装cenOS6.6 为例 如果想要需要cenos 6.6 ios文件的朋友看我的另一篇关于cenos6.6版本的下载详细 文中内容是摘抄自老男孩老师的<linux 跟老男孩学Linux运 ...
- [css 实践篇] CSS box-orient
定义和用法 box-orient 属性规定框的子元素应该被水平或垂直排列. 提示:水平框中的子元素从左向右进行显示,而垂直框的子元素从上向下进行显示.不过,box-direction 和 box-or ...
- Spring定时任务实例
一.Quartz介绍 在企业应用中,我们经常会碰到时间任务调度的需求,比如每天凌晨生成前天报表,每小时生成一次汇总数据等等.Quartz是出了名的任务调度框架,它可以与J2SE和J2EE应用程序相结合 ...
- watchdog(IWDG)
1.为了避免程序忙跑跑死了没反应,加上一个看门狗watchdog实时监控着程序,一旦程序没有在规定的时间喂狗,则狗叫使得单片机复位. 2.Independent watchdog(IWDG)内部有时钟 ...