Pots(POJ - 3414)【BFS 寻找最短路+路径输出】
时间:2022-07-26
本文章向大家介绍Pots(POJ - 3414)【BFS 寻找最短路+路径输出】,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
Pots(POJ - 3414)
算法
BFS
1.这道题问的是给你两个体积分别为A和B的容器,你对它们有三种操作,一种是装满其中一个瓶子,另一种是把其中一个瓶子的水都倒掉,还有一种就是把其中一个瓶子的水导入另一个瓶子中(可能会有剩余)。最后让你输出在能够得出体积为C的水的情况下操作的最小次数并且把过程输出。如果无法得出体积为C的水,则输出“impossible”。
2.这个题主要涉及两个点,一个是求出最小次数,还有一个就是把路径输出。对于这种有目标值的求最小次数问题,我们可以使用bfs解决。初始状态是两个瓶子都为空,最终状态是其中一个瓶子中的水的容量达到了目标值C。在每个状态下可以对瓶子进行上面描述的三种操作,细分下来其实只有6种操作,分别是:
- 将A瓶子装满水
- 将B瓶子装满水
- 将A瓶子中的水倒入B瓶子中
- 将B瓶子中的倒入A瓶子中
- 将A瓶子中的水全部抽走
- 将B瓶子中的水全部抽走
我们可以把每种状态都放入到队列中,当到达某种状态时,就分别执行上面6个操作,同时需要注意做好每种状态的标记,避免重复。
3.为了获得路径,我们可以把每次将新的状态插入队列中的同时用数组记录,可以用结构体来存放每个状态,该结构体中除了当前A瓶和B瓶中水的容量这两种属性外,还要有它本身在数组中的下标id,父状态在数组中的下标pre,到达此状态时的最小步数steps。由于bfs的特殊性,我们可以认为当达到目标值时的steps为最小最优的(至于为什么是最优的,我们可以简单的想bfs因为是一层一层的搜索的,所以可以认为第一个到达目标值的层数是最小的,当然前提是代价是一样的,比如这里每执行一步都表示一次,故可以用bfs实现,而如果执行每种操作的代价不同,就不能用bfs来实现了)。
C++代码
#include<iostream>
#include<queue>
#include<stack>
#include<cstdio>
using namespace std;
const int N = 1e6 + 10, M = 1e3;
int a, b, c;
bool st[M][M];
struct Status{
int ca, cb, pre;
int ope; //1表示装满A,2表示装满B,3表示A倒入B,4表示B倒入A,5表示倒掉A,6表示倒掉B
int steps;
int id;
Status(){
ca = 0, cb = 0, pre = -1;
ope = -1;
steps = 0;
id = 0;
}
}s[N];
int cnt;
//装满A
void fullA(Status& t)
{
t.ca = a;
t.ope = 1;
t.steps++;
t.id = cnt;
//s[cnt++] = t;
}
//倒掉A
void pourA(Status& t)
{
t.ca = 0;
t.ope = 5;
t.steps++;
t.id = cnt;
//s[cnt++] = t;
}
//装满B
void fullB(Status& t)
{
t.cb = b;
t.ope = 2;
t.steps++;
t.id = cnt;
//s[cnt++] = t;
}
//倒掉B
void pourB(Status& t)
{
t.cb = 0;
t.ope = 6;
t.steps++;
t.id = cnt;
// s[cnt++] = t;
}
//A倒入B
void pourA_B(Status& t)
{
if(t.ca >= b - t.cb)
{
t.ca -= b - t.cb;
t.cb = b;
}
else
{
t.cb += t.ca;
t.ca = 0;
}
t.ope = 3;
t.steps++;
t.id = cnt;
//s[cnt++] = t;
}
//B倒入A
void pourB_A(Status& t)
{
if(t.cb >= a - t.ca)
{
t.cb -= a - t.ca;
t.ca = a;
}
else
{
t.ca += t.cb;
t.cb = 0;
}
t.ope = 4;
t.steps++;
t.id = cnt;
//s[cnt++] = t;
}
void print(Status t)
{
cout << t.steps << endl;
stack<Status> sta;
sta.push(t);
while(true)
{
if(t.pre == -1) break;
t = s[t.pre];
sta.push(t);
}
while(sta.size())
{
Status tmp = sta.top();
sta.pop();
if(tmp.ope == 1)
puts("FILL(1)");
else if(tmp.ope == 2)
puts("FILL(2)");
else if(tmp.ope == 3)
puts("POUR(1,2)");
else if(tmp.ope == 4)
puts("POUR(2,1)");
else if(tmp.ope == 5)
puts("DROP(1)");
else if(tmp.ope == 6)
puts("DROP(2)");
}
}
void bfs()
{
Status tmp;
queue<Status> que;
que.push(tmp);
st[tmp.ca][tmp.cb] = true;
s[cnt++] = tmp;
while(que.size())
{
tmp = que.front();
que.pop();
//cout << "***" << tmp.ca << ", " << tmp.cb << endl;
if(tmp.ca == c || tmp.cb == c)
{
//cout << "**end** " << tmp.ca << " , " << tmp.cb << ", " << tmp.pre << endl;
//打印输出
print(tmp);
return ;
}
//循环那六种情况
for(int i = 1; i <= 6; i++)
{
Status tmp1 = tmp;
//cout << "tmp*** " << tmp.ca << ", " << tmp.cb << endl;
if(i == 1)
{
if(tmp1.ca == a) continue;
fullA(tmp1);
if(!st[tmp1.ca][tmp1.cb])
{
tmp1.pre = tmp.id;
st[tmp1.ca][tmp1.cb] = true;
que.push(tmp1);
s[cnt++] = tmp1;
//cout << "tmp1**1* " << tmp1.ca << ", " << tmp1.cb << endl;
}
}
else if(i == 2)
{
if(tmp1.cb == b) continue;
fullB(tmp1);
if(!st[tmp1.ca][tmp1.cb])
{
tmp1.pre = tmp.id;
st[tmp1.ca][tmp1.cb] = true;
que.push(tmp1);
s[cnt++] = tmp1;
//cout << "tmp1**2* " << tmp1.ca << ", " << tmp1.cb << endl;
}
}
else if(i == 3)
{
if(!(tmp1.ca)) continue;
pourA_B(tmp1);
if(!st[tmp1.ca][tmp1.cb])
{
tmp1.pre = tmp.id;
st[tmp1.ca][tmp1.cb] = true;
que.push(tmp1);
s[cnt++] = tmp1;
//cout << "tmp1**3* " << tmp1.ca << ", " << tmp1.cb << endl;
}
}
else if(i == 4)
{
if(!(tmp1.cb)) continue;
pourB_A(tmp1);
if(!st[tmp1.ca][tmp1.cb])
{
tmp1.pre = tmp.id;
st[tmp1.ca][tmp1.cb] = true;
que.push(tmp1);
s[cnt++] = tmp1;
//cout << "tmp1**4* " << tmp1.ca << ", " << tmp1.cb << endl;
}
}
else if(i == 5)
{
if(!(tmp1.ca)) continue;
pourA(tmp1);
if(!st[tmp1.ca][tmp1.cb])
{
tmp1.pre = tmp.id;
st[tmp1.ca][tmp1.cb] = true;
que.push(tmp1);
s[cnt++] = tmp1;
//cout << "tmp1**5* " << tmp1.ca << ", " << tmp1.cb << endl;
}
}
else if(i == 6)
{
if(!(tmp1.cb)) continue;
pourB(tmp1);
if(!st[tmp1.ca][tmp1.cb])
{
tmp1.pre = tmp.id;
st[tmp1.ca][tmp1.cb] = true;
que.push(tmp1);
s[cnt++] = tmp1;
//cout << "tmp1**6* " << tmp1.ca << ", " << tmp1.cb << endl;
}
}
}
}
puts("impossible");
}
int main()
{
cin >> a >> b >> c;
bfs();
return 0;
}
- UVA 1030 - Image Is Everything【模拟+思维+迭代更新】
- C/C++中对链表操作的理解&&实例分析
- COGS 2482. Franky的胡子【二分,高精度】
- 【AlphaGo Zero 核心技术-深度强化学习教程代码实战02】理解gym的建模思想
- Kruscal(最小生成树)算法模版
- BZOJ 3680: 吊打XXX【模拟退火算法裸题学习,爬山算法学习】
- 区块链概况:什么是比特币
- 简化版桶排序操作模版
- BZOJ 3670: [Noi2014]动物园【KMP变形 】
- 【延迟注入】A5站长网某站存在SQL注入漏洞
- hihoCoder #1498 : Diligent Robots【数学】
- Codeforces Round #336 (Div. 2)【A.思维,暴力,B.字符串,暴搜,前缀和,C.暴力,D,区间dp,E,字符串,数学】
- 【防护】如何阻止SELECT * 语句
- COGS 862. 二进制数01串【dp+经典二分+字符串】
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- three.js 数学方法之Matrix4
- three.js 利用uv和ThreeBSP制作一个快递柜
- three.js 欧拉角和四元数
- three.js 制作逻辑转体游戏(上)
- three.js 制作魔方
- 安卓日常开发和逆向中常用的shell命令与非shell命令
- 使用python生成Latex公式语法
- 设计模式~桥梁模式
- 设计模式~模板方法模式
- three.js 郭先生制作太阳系
- 视频监控系统视频上云解决方案EasyCVR集成海康EHome私有协议系列——设备录像回看说明
- 视频监控系统视频上云解决方案EasyCVR集成海康EHome私有协议系列——报警模块说明
- three.js 制作一个三维的推箱子游戏
- IP摄像头RTSP协议网页无插件直播视频平台EasyNVR如何在直播或录像中添加定制化音频?
- three.js之初探骨骼动画