题目如下:

21-三个水杯

内存限制:64MB 时间限制:1000ms 特判: No
通过数:51 提交数:137 难度:4

题目描述:

给出三个水杯,大小不一,并且只有最大的水杯的水是装满的,其余两个为空杯子。三个水杯之间相互倒水,并且水杯没有标识,只能根据给出的水杯体积来计算。现在要求你写出一个程序,使其输出使初始状态到达目标状态的最少次数。

输入描述:

第一行一个整数N(0<N<50)表示N组测试数据
接下来每组测试数据有两行,第一行给出三个整数V1 V2 V3 (V1>V2>V3 V1<100 V3>0)表示三个水杯的体积。
第二行给出三个整数E1 E2 E3 (体积小于等于相应水杯体积)表示我们需要的最终状态

输出描述:

每行输出相应测试数据最少的倒水次数。如果达不到目标状态输出-1

样例输入:

复制

2
6 3 1
4 1 1
9 3 2
7 1 1

样例输出:

3
-1

解题思想:题目不难,主要是捉住关键点就容易了,看到最少的倒水次数,必然和广度优先搜索有联系,其实就是一颗六叉树。

  每个杯子都可以往其他两个杯子倒水。可以相互倒水,情况如下:1>2,1>3,2>1,2>3,3>1,3>2;

代码如下:用到了队列+广度的方法。

import java.util.Arrays;
import java.util.LinkedList;
import java.util.Queue;
import java.util.Scanner; class State
{
int glass[] = new int[3];
int step;
}
public class Main
{
static final int MAX = 3;
static int v[] = new int[MAX];
static int e[] = new int[MAX];
static boolean vis[][][] = new boolean[300][300][300];
static int flag;
public static void main(String []args)
{
Scanner cin = new Scanner(System.in);
int N = cin.nextInt();
for(int i = 0; i < N; i++)
{
for(int j = 0; j < 3; j++)
{
v[j] = cin.nextInt();
}
for(int j = 0; j < 3; j++)
{
e[j] = cin.nextInt();
}
flag = 0;
Init();
BFS();
if(flag == 0)
{
System.out.println("-1");
}
}
}
static void Init()
{
for(int i = 0; i < v[0]; i++)
{
for(int j = 0; j < v[0]; j++)
{
for(int k = 0; k < v[0]; k++)
{
vis[i][j][k] = false;
}
}
}
}
static void BFS()
{
State start = new State();
start.glass[0] = v[0];
start.glass[1] = 0;
start.glass[2] = 0;
if(start.glass[0] == e[0] && start.glass[1] == e[1] && start.glass[2] == e[2])
{
flag = 1;
System.out.println("0");
return;
}
Queue <State> que = new LinkedList<State>();
que.add(start);
vis[start.glass[0]][start.glass[1]][start.glass[2]] = true;
while(!que.isEmpty())
{
State temp = new State();
temp = que.poll();
for(int i = 0; i < 3; i++)//核心代码
{
if(temp.glass[i] == 0)
{
continue;
}
//i向j中倒水
for(int j = 0; j < 3; j++)
{
if(j == i || temp.glass[j] == v[j]) //同一个杯子或者这个杯子已经满了
{
continue;
}
//向其他某个杯子倒
State mid = new State();//此处的对象用得很巧妙(不可以定义在这个循环的外面),我之前都是建立一个很大很大的数组用来保存队列,太浪费空间了。
mid.glass[0] = temp.glass[0];
mid.glass[1] = temp.glass[1];
mid.glass[2] = temp.glass[2];
mid.step = temp.step+1;
int rest = v[j]-temp.glass[j];
if(rest <= mid.glass[i])
{
mid.glass[j] = v[j];
mid.glass[i] = mid.glass[i]-rest;
if(vis[mid.glass[0]][mid.glass[1]][mid.glass[2]] == false)
{
if(mid.glass[0] == e[0] && mid.glass[1] == e[1] && mid.glass[2] == e[2])
{
flag = 1;
System.out.println(mid.step);
return;
}
que.add(mid);
vis[mid.glass[0]][mid.glass[1]][mid.glass[2]] = true;
}
}
else
{
mid.glass[j] = mid.glass[j] + mid.glass[i];
mid.glass[i] = 0;
if(vis[mid.glass[0]][mid.glass[1]][mid.glass[2]] == false)
{
if(mid.glass[0] == e[0] && mid.glass[1] == e[1] && mid.glass[2] == e[2])
{
flag = 1;
System.out.println(mid.step);
return;
}
que.add(mid);
vis[mid.glass[0]][mid.glass[1]][mid.glass[2]] = true;
}
}
}
}
}
}
}

三个水杯——java,广度优先搜索的更多相关文章

  1. BFS(三):双向广度优先搜索

    所谓双向广度搜索指的是搜索沿两个方向同时进行:(1)正向搜索:从初始结点向目标结点方向搜索:(2)逆向搜索:从目标结点向初始结点方向搜索:当两个方向的搜索生成同一子结点时终止此搜索过程. 广度双向搜索 ...

  2. 利用广度优先搜索(BFS)与深度优先搜索(DFS)实现岛屿个数的问题(java)

    需要说明一点,要成功运行本贴代码,需要重新复制我第一篇随笔<简单的循环队列>代码(版本有更新). 进入今天的主题. 今天这篇文章主要探讨广度优先搜索(BFS)结合队列和深度优先搜索(DFS ...

  3. java深度搜索与广度优先搜索

    图结构展示: 实现过程: 首先,我们来看看图结构在代码中的实现.有三块逻辑: 1.图中的节点: public class GraphNode { public List<GraphEdge> ...

  4. 深度优先搜索(DFS)与广度优先搜索(BFS)的Java实现

    1.基础部分 在图中实现最基本的操作之一就是搜索从一个指定顶点可以到达哪些顶点,比如从武汉出发的高铁可以到达哪些城市,一些城市可以直达,一些城市不能直达.现在有一份全国高铁模拟图,要从某个城市(顶点) ...

  5. Java用广度优先搜索快速搜索文件

    背景 在开发的过程中,经常需要在文件系统里按某些条件搜索文件,比如音乐播放器扫描音乐,而搜索文件,大多人喜欢用递归的方式,而这也是最容易想到的方式.递归方式如果文件夹很深就容易造成栈溢出,而且不断的压 ...

  6. Java爬虫搜索原理实现

    permike 原文 Java爬虫搜索原理实现 没事做,又研究了一下爬虫搜索,两三天时间总算是把原理闹的差不多了,基本实现了爬虫搜索的原理,本次实现还是俩程序,分别是按广度优先和深度优先完成的,广度优 ...

  7. nyoj 题目21 三个水杯

    三个水杯 时间限制:1000 ms  |  内存限制:65535 KB 难度:4   描述 给出三个水杯,大小不一,并且只有最大的水杯的水是装满的,其余两个为空杯子.三个水杯之间相互倒水,并且水杯没有 ...

  8. Leetcode之广度优先搜索(BFS)专题-详解429. N叉树的层序遍历(N-ary Tree Level Order Traversal)

    Leetcode之广度优先搜索(BFS)专题-429. N叉树的层序遍历(N-ary Tree Level Order Traversal) 给定一个 N 叉树,返回其节点值的层序遍历. (即从左到右 ...

  9. NYOJ 21 三个水杯

    三个水杯 时间限制:1000 ms  |  内存限制:65535 KB 难度:4   描述 给出三个水杯,大小不一,并且只有最大的水杯的水是装满的,其余两个为空杯子.三个水杯之间相互倒水,并且水杯没有 ...

随机推荐

  1. 一道有趣的JS问题

    function Foo() { getName = function () { alert (1); }; return this; } Foo.getName = function () { al ...

  2. 控制台出现“The script has an unsupported MIME type ('text/html')”报错

    有时候开发React或者Vue项目时,本地运行访问时,会莫名出现报错如下: The script has an unsupported MIME type ('text/html') 这是由于无意中开 ...

  3. work-7.2

    安装ubuntu,jdk ,git,maven,Intellij. 配置GIT时,需要将在本地生成的公钥粘贴到服务端. 先占个座,具体过程待补充. -------------------------- ...

  4. oracle删除数据库

    1.确认当前数据库是否为要删除的那一个select name from v$database; 2.关闭数据库shutdown immediate; 3.以restrict方式重新打开数据库,并启动到 ...

  5. ok6410 nandflash 启动uboot 超过256k怎么办

    1\在调试ok6410 nandflash启动uboot时,需要添加调试信息,导致uboot大于256k 2\需要修改下面几个地方 参考资料 在uboot中可以添加文字菜单,但是不够美观. 利用pho ...

  6. 用vue脚手架创建bootstrap-vue项目

    用vue脚手架创建bootstrap-vue项目 框架的地址:https://bootstrap-vue.js.org/docs/ 第一步 vue init webpack demo第二步 cd de ...

  7. 基于ROS的人脸识别

    #!/usr/bin/env python # -*- coding: utf-8 -*- import rospy import cv2 import numpy as np from sensor ...

  8. First Python script

    learn what is api Jailbreak pycharm install requests on pycharm write first request: get, post MFA l ...

  9. es2018(es9)前瞻

    命名捕获 语法 : ?<name> 一:举个栗子 我们要把从2018-05-20取出年月日 1:普通方法 let str = '2018-05-20'; let reg1 = /(\d{4 ...

  10. centos6添加mysql服务自启动

    环境: os: centos 6 db:mysql 5.6.40 1.修改/etc/init.d/mysqld文件,默认文件以及存在,该文件内容如下: #!/bin/sh # Copyright Ab ...