蓝桥杯第642题——跳蚱蜢

news2025/4/27 22:07:56

题目描述

如下图所示: 有 9 只盘子,排成 1 个圆圈。 其中 8 只盘子内装着 8 只蚱蜢,有一个是空盘。 我们把这些蚱蜢顺时针编号为 1 ~ 8。

图片描述

每只蚱蜢都可以跳到相邻的空盘中, 也可以再用点力,越过一个相邻的蚱蜢跳到空盘中。

请你计算一下,如果要使得蚱蜢们的队形改为按照逆时针排列, 并且保持空盘的位置不变(也就是 1−8 换位,2−7换位,...),至少要经过多少次跳跃?

解题思路

这道题是典型的BFS相关例题,可以练习BFS判重,以及双向广搜的具体方式。

我们从蚱蜢跳跃去考虑并不方便,因为我们每次都有很多只蚱蜢可以进行跳跃,因此我们可以转换思维,考虑到只有一个空盘子,所以第一次跳跃的情况是必定有一只蚱蜢跳到空盘子里;而下一次操作也是同样的情况。

所以我们考虑有哪些蚱蜢可以跳到空盘子上,进一步思考,就是空盘子可以与哪些蚱蜢交换位置,很明显,如图所示,0号空盘可以与2、1、8、7号蚱蜢交换位置,这就为我们提供了搜索路径的灵感;同时,我们是要找到最少的跳跃次数,也就是最短路径,因此我们考虑广度优先搜索。

因为这题不像迷宫,有一个明显的特点,也就是0和2交换的结果——可以由0和1交换,0和2交换,1再和2交换获得。那么这就会导致很多重复,所以后来变换的结果如果与前面已经获取过的结果有重叠,就应当剪枝不再计算,这里我们考虑使用HashSet结构和HashMap结构进行去重。

单点BFS

我们使用一个ArrayDeque构成的队列实现BFS,队列中每个节点的内容是当前的字符串s和得到它所走过的步数step,另外使用一个HashSet来记录搜索过的字符串,很明显先进来的字符串所带来的step数值是最小的,判重过后只需要忽略后续来的相同字符串即可。

以下是常规bfs判重的解题代码:

import java.io.*;
import java.util.*;

public class Main {
    static String S1 = "012345678";
    static String S2 = "087654321";
    static int num = 0;     // 计算队列的操作次数,用于与双向广搜对比
    public static void main(String[] args) throws IOException {
         Set<String> set = new HashSet<>();
         ArrayDeque<Pair> qu = new ArrayDeque<>();
         qu.addLast(new Pair(S1, 0));
         set.add(S1);
         while (!qu.isEmpty()) {
             Pair temp = qu.removeFirst();
             if (temp.s.equals(S2)) {
                 System.out.println(temp.step);
                 System.out.println(num);
                 break;
             }
             // 使用index记录当前字符串中0所在的位置
             int index = 0;
             for (int i = 0; i < 9; i++) {
                 if (temp.s.charAt(i) == '0') {
                     index = i;
                     break;
                 }
             }
             for (int i = index - 2; i <= index + 2; i++) {
                 num++;     // num的每次自增意味着计算的执行次数增加
                 // 使用取余技巧获取将要进行交换的下标
                 int k = (i + 9) % 9;
                 // 5次循环中忽略实际没有交换的情况
                 if (k == index) {
                     continue;
                 }
                 char[] ss = temp.s.toCharArray();
                 char t = ss[k]; ss[k] = ss[index]; ss[index] = t;
                 if (set.add(String.copyValueOf(ss))) {
                     qu.addLast(new Pair(String.copyValueOf(ss), temp.step + 1));
                 }
             }
         }
    }
}

class Pair {
    String s;
    int step;

    public Pair(String s, int step) {
        this.s = s;
        this.step = step;
    }
}

 结果输出中,第一个数值是题目所要求的答案,第二个值代表着该算法的运行次数。

20
1814315
第一种思路的局限性

在考虑单点bfs找最短路径的情况,我们可以想象原点是一颗由上至下的4叉树,那么题目的答案20,意味着我们要向下分出20层才能找到答案,树会不断往底部扩展,越是向下的层数,所需要计算的节点就越是多,如果我们没有去重,想象一个完整的20层4叉树,它是非常巨大的。

那么我们可以考虑另外一种情况,现在我们明确的知道目标结果的字符串,那么我们可以不可以根据最终状态再反向生成一颗树呢?

如果从上至下以及从下至上同时开始广搜,那么我们所需要生成的四叉树就从一个20层的树变成了两个10层的树,同学们可以自行想象,这对时间的节省又是相当巨大的。

双向广搜

在上述单点BFS的基础上,我们进行升级为双向广搜,每次对队列的一次BFS操作都根据队列长度较短的进行选择执行,第一次遇到相同字符串的时候,两个层数加起来就是我们的答案。

我们需要将第一种方式的一个Set修改为两个独立的HashMap,记录字符串出现的同时,也要对应记录其出现的层数,方便我们对答案进行输出。

import java.io.*;
import java.util.*;

public class Main {
    static String S1 = "012345678";
    static String S2 = "087654321";
    static int times = 0;
    static int ans = 0;
    public static void main(String[] args) throws IOException {
        HashMap<String, Integer> map1 = new HashMap<>();
        HashMap<String, Integer> map2 = new HashMap<>();
        ArrayDeque<Pair> qu1 = new ArrayDeque<>();
        ArrayDeque<Pair> qu2 = new ArrayDeque<>();
        qu1.addLast(new Pair(S1, 0));
        map1.put(S1, 0);
        qu2.addLast(new Pair(S2, 0));
        map2.put(S2, 0);

        while (!qu1.isEmpty() && !qu2.isEmpty()) {
            if (qu1.size() < qu2.size()) {
                extend(qu1, map1, map2);
            } else {
                extend(qu2, map2, map1);
            }
            if (ans != 0) {
                System.out.println(ans);
                System.out.println(times);
                break;
            }
        }
    }
    // 注意该方法中的qu是当前操作的队列
    // map对应的是qu的HashMap
    // 而map1是另外一个队列的HashMap
    public static void extend(ArrayDeque<Pair> qu, HashMap<String, Integer> map, HashMap<String, Integer> map1) {
        Pair temp = qu.removeFirst();
        if (map1.containsKey(temp.s)) {
            ans = temp.step + map1.get(temp.s);
            return;
        }
        int index = -1;
        for (int i = 0; i < temp.s.length(); i++) {
            if (temp.s.charAt(i) == '0') {
                index = i;
                break;
            }
        }
        for (int i = index - 2; i <= index + 2; i++) {
            times++;
            int k = (i + temp.s.length()) % temp.s.length();
            if (k == index) {
                continue;
            }
            char[] arr = temp.s.toCharArray();
            char ch = arr[index]; arr[index] = arr[k]; arr[k] = ch;
            String news = String.copyValueOf(arr);
            if (!map.containsKey(news)) {
                map.put(news, temp.step + 1);
                qu.addLast(new Pair(news, temp.step + 1));
            }
        }
    }
}

class Pair {
    String s;
    int step;

    public Pair(String s, int step) {
        this.s = s;
        this.step = step;
    }
}
20
133510

通过两种方法的输出对比我们可以看出,对队列的操作次数由180万降低到了13万,这也是双向广搜的厉害之处。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1529175.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Maven入门篇】(3)依赖配置,依赖传递,依赖范围,生命周期

&#x1f38a;专栏【Maven入门篇】 &#xfeff;> &#x1f354;喜欢的诗句&#xff1a;更喜岷山千里雪 三军过后尽开颜。 &#xfeff;> &#x1f386;音乐分享【The truth that you leave】 &#xfeff;> &#x1f970;欢迎并且感谢大家指出我的问题 文章目录 &…

Spring之@Value注解

前言 Value注解在Spring的依赖注入中占据重要地位,这里对Value注解的作用进行演示以及扩展 作用 注入字符串注入属性注入bean其他 代码准备 创建两个普通的bean Component public class ValueComponent { } Component public class Foo {private String sign;public Foo…

算法——位运算(一篇搞定)

本专栏为大家分享本人学习算法遇到的不同类型的题目以及解析! 此篇文章给大家分享一些关于位运算算法的题目,在开篇还讲述了常见位运算的公式以及题目,如果对您有帮助,麻烦点个关注,如有错误,请您指出! 1.常见位运算总结(包含5道题目) 1.1基础位运算 运算符操作<<左移&…

Google XSS Game Level 6 通关方式

文章目录 链接&#xff1a;[Google XSS Game](#https://xss-game.appspot.com/)Level 6 - Follow the &#x1f407;思路1 &#xff08;当然&#xff0c;我使用这个方式没有成功&#xff0c;所以才来记录下&#xff09;解法2 【最简单的解法】需要注意的一个小问题 链接&#x…

卡特兰数的部分解析(1)

直接开始 关于卡特兰数有三个公式 这里会解析一下第二个公式的含义 直接上公式 C n C 2 n n − C 2 n n − 1 或者是 C n C 2 n n − C 2 n n 1 C_n C_{2n}^n - C_{2n}^{n-1} 或者是 C_n C_{2n}^n - C_{2n}^{n1} Cn​C2nn​−C2nn−1​或者是Cn​C2nn​−C2nn1​ 解析…

网络编程—DAY5

select实现的TCP并发服务器 #include <myhead.h> #define SER_PORT 8888 #define SER_IP "192.168.117.96"int main(int argc, const char *argv[]) {int sfd -1;sfd socket(AF_INET,SOCK_STREAM,0);if(sfd -1){perror("socket");return -1;}prin…

Python文件操作相关知识点(读取/写入数据)

1.open函数的定义 &#xff08;1&#xff09;open函数的简要概述 open&#xff08;filename&#xff09;函数接受一个参数——要打开文件的名称&#xff0c;Python在当前执行的文件所在的目录中查找指定的文件。并返回一个表示文件的对象。 open&#xff08;&#xff09;函数…

基于python智慧社区家政服务系统的设计与实现flask-django-nodejs-php

随着现代网络技术发展&#xff0c;对于智慧社区家政服务系统的设计现在正处于发展的阶段&#xff0c;所以对的要求也是比较严格的&#xff0c;要从系统的功能和用户实际需求来进行对系统制定开发的发展方式&#xff0c;依靠网络技术的的快速发展和现代通讯技术的结合为人们带来…

MyBatis是纸老虎吗?(四)

在《MyBatis是纸老虎吗&#xff1f;&#xff08;三&#xff09;》这篇文章中我们一起梳理了MyBatis配置文件的解析流程&#xff0c;并详细介绍了其中的一些常见节点的解析步骤。通过梳理&#xff0c;我们弄清楚了MyBatis配置文件中的一些常用配置项与Java Bean之间的对应关系&a…

threejs之贴图原理

// 导入threejs import * as THREE from "three"; // 导入轨道控制器 import { OrbitControls } from "three/examples/jsm/controls/OrbitControls.js";// 创建场景 const scene new THREE.Scene();// 创建相机 const camera new THREE.PerspectiveCame…

使用Java JDBC连接数据库

在Java应用程序中&#xff0c;与数据库交互是一个常见的任务。Java数据库连接&#xff08;JDBC&#xff09;是一种用于在Java应用程序和数据库之间建立连接并执行SQL查询的标准API。通过JDBC&#xff0c;您可以轻松地执行各种数据库操作&#xff0c;如插入、更新、删除和查询数…

有没搞错!花了大价钱的激光孔设计性能竟然不如普通通孔?

高速先生成员--黄刚 老话说得好&#xff0c;一分耕耘一分收获&#xff0c;又或者另外一句&#xff0c;有什么付出就会得到多少收获。我们都不会去怀疑这些话的正确性。但是把这两句话用到PCB领域中&#xff0c;用了好的加工工艺后&#xff0c;PCB板的性能就一定会比用普通工艺要…

视频素材免费下载素材库哪里有?推荐8个高清无水印素材网

在这个数字化时代&#xff0c;无论是专业的内容创作者还是日常的社交媒体使用者&#xff0c;我们都会寻找高质量的素材来丰富我们的作品或帖子。从令人震撼的摄影作品到高分辨率的视频素材&#xff0c;再到生动的GIF和必需的设计元素&#xff0c;素材的需求无处不在。 视频素材…

XXL-JOB完全开发手册(一篇学会XXL-JOB所有知识点)

目录 1、什么是XXL-JOB 1.1、XXL-JOB简介 1.2、XXL-JOB构成 调度模块&#xff08;调度中心&#xff09;&#xff1a; 执行模块&#xff08;执行器&#xff09;&#xff1a; 任务&#xff1a; 1.3、XXL-JOB总结 ​编辑 2、XXL-JOB原理 2.1、执行器的注册和发现 2.2、调度中心调用…

基于华为ensp的企业网络规划(新版)

第一章 项目概述 1.1 项目总体描述 假设某大型公司总部在北京、在重庆设置分部&#xff0c;总部和分部均有研发部、市场部、财务部等部门&#xff0c;现在要求进行网络规划与设计&#xff0c;实现分部和总部能够进行网络连通。为了保证数据安全&#xff0c;在总部和分部之间可…

教师怎么发成绩才不会被投诉

在当今社会&#xff0c;教育的重要性日益凸显&#xff0c;而学生的成绩作为衡量教育效果的重要指标之一&#xff0c;备受家长和学生的关注。然而&#xff0c;教师在发布成绩时&#xff0c;稍有不慎就可能引发家长的不满和投诉。那么&#xff0c;教师该如何发成绩才能避免被投诉…

【Spring Cloud Gateway】路由配置uri三种方式及区别

websocket配置方式 ws:// 或 wss:// 开头的 URI&#xff0c;表示配置的是支持 Websocket 协议的目标地址。 这种方式适用于需要与客户端建立长连接、实现双向通信的场景&#xff0c;比如实时消息推送、即时聊天等。 使用 Websocket 配置方式可以让 Spring Cloud Gateway 能够…

【物联网】Modbus 协议及应用

Modbus 协议简介 QingHub设计器在设计物联网数据采集时不可避免的需要针对Modbus协议的设备做相关数据采集&#xff0c;这里就我们的实际项目经验分享Modbus协议 简介 Modbus由MODICON公司于1979年开发&#xff0c;是一种工业现场总线协议标准。1996年施耐德公司推出基于以太…

虚拟机VMware上 centos7 的网络配置

第一步&#xff1a;权限的切换 由普通用户切换到管理者/超级用户 用户名为&#xff1a;root 密码为&#xff1a;自己安装 linux 时第一次设置的密码 su -root管理者/超级用户的命令提示符是“#”&#xff0c;普通用户的命令提示符是“$”。当看到你的命令提示符为“$”时&…

《世界之外》玩家闹上315,乙游打响维权大战

315维权微博的评论区&#xff0c;竟然被举报网易的玩家占领了。 玩家举报网易乙游《世界之外》虚假宣传侵害消费者权益&#xff0c;在游戏中设置排行榜和专属商店将玩家分为三六九等&#xff0c;诱导玩家消费氪金&#xff0c;强烈要求网易打开退款通道。 目前大批玩家举报的举…