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 ...
随机推荐
- MyBatis_Generator插件的安装以及简单使用
MyBatis_Generator_1.3.1.zip 1 下载安装包 安装包名称:MyBatis_Generator_1.3.1.zip 2 在Eclipse上进行安装 l ...
- jquery attr处理checkbox / select 等表单元素时的坑
先上html结构 <body> <form action=""> <input type="checkbox" id=" ...
- JavaSE教程-04Java中循环语句for,while,do···while
** Java的循环语句 ** 引入: 生活中有循环,程序的世界也有循环. 思考:生活中有哪些循环的事情? 总结:什么是循环? 重复做类似的事情,而且有终止条件,如果没有终止条件会是怎样? 类似这样的 ...
- ActionContext、ServletContext、pageContext的区别?
ActionContext是当前的Action的上下文环境,通过ActionContext可以获取到request.session.ServletContext等与Action有关的对象的引用: Se ...
- 关于XAMPP环境配置
关于XAMPP软件 * Apache - 软件服务器(运行PHP) * 启动失败 * 原因 - 端口号被占用 * 错误信息 - Error: Apache shutdown unexpectedly ...
- iptables实用教程(一):基本概念和原理
概述 iptables是linux自带的防火墙软件,用于配置IPv4数据包过滤或NAT(IPv6用ip6tables). 在linux上,防火墙其实是系统内核的一部分,基于Netfilter构架,基本 ...
- 【Python3之面向对象的程序设计】
一.面向对象的程序设计的由来 1.第一阶段:面向机器,1940年以前 最早的程序设计都是采用机器语言来编写的,直接使用二进制码来表示机器能够识别和执行的指令和数据. 简单来说,就是直接编写 0 和 1 ...
- Aop初步了解
AOP(aspect-oriented programming) aspect是一种新型的模块化机制,用来描述分散在对象,类或函数中横切关注点.从关注点中分离出横切关注点是面向切面的程序设计的核心概念 ...
- 破解 Adobe 系列的最佳方法,手把手教
此方法是个人认为最方便的而且最安全的方法(可以避免下载到可能捆绑病毒的破解版本) 1.首先到Adobe的官网上下载 Creative Cloud: 打开官网上creative cloud 的下载页面: ...
- mac os 安装PIP 及异常“”Can't install python module: PyCharm Error: “byte-compiling is disabled, skipping”“”的解决方案
For all who have the same problem, it took me a while to find the solution in a new installation of ...