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 ...
随机推荐
- 容器扩展属性 IExtenderProvider 实现WinForm通用数据验证组件
大家对如下的Tip组件使用应该不陌生,要想让窗体上的控件使用ToolTip功能,只需要拖动一个ToolTip组件到窗口,所有的控件就可以使用该功能,做信息提示. 本博文要记录的,就是通过容器扩展属性 ...
- void main(), int main() 和int main(void)的区别
1.区别是main()函数是否有返回值.2.void定义的函数没有返回值,int定义的函数返回整型值.3.void,字面意思是"无类型",常用在程序编写中对定义函数的参数类型.返回 ...
- VR全景:电商巨头的角逐
VR全景智慧城市:京东推"京东梦"挑战淘宝Buy+ ,VR购物谁主沉浮? VR全景智慧城市是国内首家商业全景平台,结合先进VR虚拟现实技术,以线下实体为依托,将空间还原到线上,用户 ...
- FFmpeg安装(windows环境)
♣FFmpeg是什么? ♣FFmpeg组成 ♣下载工具 ♣安装FFmpeg ♣应用到j2ee项目 前言:学习视频编码,一定要知道雷霄骅(leixiaohua1020)的专栏 ,伟大的程序员,26岁去世 ...
- {网络编程}和{多线程}应用:基于UDP协议【实现多发送方发送数据到同一个接收者】--练习
要求: 使用多线程实现多发送方发送数据到同一个接收者 代码: 发送端:可以在多台电脑上启动发送端,同时向接收端发送数据 注意:匹配地址和端口号 package com.qf.demo; import ...
- dedecms做好的网站怎么上传到网上?
1.首先做好网站后把网站所有和数据库备份 dedecms 点击 系统 - 数据库备份/还原 - 全选 后---提交-----等待备份完全 备份文件在哪里:data/backupadta--- 2. ...
- php实现ppt转图片,php调用com组件问题
PHP 调用com组件将ppt转为图片. 需要在php.ini中开启 extension=php_com_dotnet.dllcom.allow_dcom = true 测试代码如下: < ...
- Step by Step 用Azure Automation 来开虚机(ARM)
使用Azure Automation来自动化处理各种重复的耗时的云管理任务从而帮助云运维人员提升效率,帮助降低运营成本. 具体相关的介绍以及怎样利用Azure Automation来完成定期开关虚拟机 ...
- javascript痛点之四this的指向问题
先看以下例子 1.我们直接调用this看看指向的是谁 alert(this);//指向window 2.在函数中直接调用看看指向的是谁 function fn(){ alert(this); } fn ...
- od的用法
一.功能 查看非文本文档的内容 二.用法 1.语法 od [-A RADIX] [-t TYPE] 文件名 2.选项与参数 -A: 指定偏移量的输出形式 d[size] :利用十进制(decimal) ...