赞
踩
2024/03/19:程序更新说明(文末程序下载链接已更新)
版本:v1.0 → v1.2
① 修复:将 CLOSED 表内容从优先级队列中分离开来,原优先级队列作 OPEN 表,并用链表树隐式地代替 CLOSED 表,以此修复之前版本内存爆炸的问题(尤其以 HC 爬山法严重);
② 优化:实时搜索树动态绘制,而非总是保持 10 行;
③ 其它:优化代码,修复其它的一些小问题;
许久没有写图形化界面的程序了,最近学习了一些经典的盲目搜索算法与智能搜索算法,正好拿来还原三阶魔方!试试手!
提前声明
我不是专业搞人工智能的,理论或者实现过程有些许错误也很正常,评论区指出即可
先说一下开发环境吧!源码及打包程序的下载链接放在文末!
编程语言:Python 3.12
图形界面库:tkintertools 2.6.21.1(底层为 tkinter)
第三方依赖库:tkintertools 2.6.21.1(就这么一个)
代码量:1000 行左右
代码大小:34KB
左侧是 3D 显示区,鼠标左键旋转,右键平移,滚轮缩放。
右侧是设定区,点击“开始搜索并还原”时会弹出搜索树弹窗,点击“随机打乱”左边的“/”会弹出“自定义打乱顺序”弹窗。
算法对应表:
缩写 | BFS | DFS | UCS | A / A* | HC | REV(测试常用) |
说明 | 宽度优先 | 深度优先 | 代价优先 | A 或者 A* 算法 | 爬山法 | 不是算法,逆序还原 |
启发函数对应表:
缩写 | CBSV | ECLD | MHT | HM | MKWSK | / |
说明 | 切比雪夫距离 | 欧几里得距离 | 曼哈顿距离 | 汉明距离 | 闵可夫斯基距离 | h* |
自定义顺序说明:
每个项目由三个部分组成:编号,方向,旋向
方向对应表:
缩写 | R | L | U | D | F | B |
说明 | 右(Right) | 左(Left) | 上(Up) | 下(Down) | 前(Front) | 后(Back) |
旋向有两个:顺时针和逆时针,对应开关的两种状态
三阶魔方一共有 3³ =27 个方块,于是使用 1×27 大小的数组来表示每个方块的位置,给它们编号 0~26,当编号与其数组索引一致时,表示魔方为还原状态。
当然,由于魔方的特性,这 27 个位置中有些并不会改变。比如,当操作不涉及中转的时候,有 1+1×6 = 7 个方块永远不会改变位置,而当操作涉及中转的时候,只有中心处方块不会改变位置。为了方便表示,仍然采用整个 1×27 大小的数组表示状态。
分两种情况:
当然,经分析,我认为采用第 1 种情况可能会得到更好结果。
对于第一种情况,魔方将有 1+1×6 = 7 个位置始终固定,使得在完成几乎相同功能的前提下,搜索空间会小一点,且,还原的最终结果只有1个,那就是数组元素与其索引一致。
但反观第二种情况,魔方只有最中间 1 个元素始终固定,在使得搜索空间变大的情况下,还会导致一个程序中不方便处理的问题。因为在这个时候,你会发现,魔方还原的目标状态不只“数组元素与其索引一致”这么一种情况,虽然这可以提高发现目标节点的概率,但搜索空间也变大了,程序实现起来比较麻烦,效率也不见得比第 1 种好。
魔方数据在数组中体现为 1×27,并不方便于直接进行代价的估计,但其在空间上实际是 3×3×3 的,每个方块都有其对应的坐标,于是可以计算每个方块当前位置与目标位置之间的某种差异,并以此作为估计值。
选用的基本启发函数有切比雪夫距离、欧几里得距离、曼哈顿距离和闵可夫斯基距离,同时尝试了一下汉明距离。
不知道什么是切比雪夫距离、欧几里得距离等的朋友,可以去百度一下。
设上述距离对应的启发函数分别为 、、、 和 。
其中闵可夫斯基距离拥有可调节的参数 p,我这里动态地根据问题的规模大小设计其值。而汉明距离并非一个空间上的关系,它是从数组的数据上直接进行考虑的,即目标数组与当前数组的差异。
对于上述的启发函数,并非所有都满足可纳性。由于魔方转动一次只能转动一个面,也就是说,方块只能在一个面上移动,对于方块,分两种情况:
每个面上角方块与边方块各占一半,故 ,但是有:
因此有:
已知,对于闵可夫斯基距离,参数 p=1 时,,参数 p=2 时,,参数 p=+∞ 时,,可通过调控其参数 p 来控制其最终效果。
汉明距离并非空间上的距离,属于抽象距离,无法与上面的进行比较。
综上,启发函数为切比雪夫距离、欧几里得距离时,算法为 A* 算法,曼哈顿距离对应的为 A 算法,闵可夫斯基距离是否为 A* 算法与参数 p 有关,汉明距离对应的暂时无法确定。
BFS:用队列实现
DFS:用堆栈实现
UCS:用优先级队列实现,评估函数 F = 代价函数 G
A/A*:用优先级队列实现,评估函数 F = 代价函数 G + 启发函数 H
HC:用优先级队列实现,评估函数 F = 启发函数 H
结果采用三种方式进行可视化。
这里说一下为什么实际搜索的状态空间是 11 的 n 次方,而不是 12 的 n 次方,因为每次操作,虽然有 12 步,但实际我们手动不允许它执行与上次转动相反的操作,因为你顺时针转一下,再逆时针转一下,那不等于没转吗?
别小看这点优化,11⁶ = 1771561,12⁶ = 2985984,仅 6 步,相差多少自己看看。
本程序使用的 tkintertools 是我自己一个人开发的图形界面库,基于 tkinter,实现了一些 tkinter 没有的功能,里面还有一个可以称为“微型 3D 引擎”的子模块,上述 3D 效果就是这样实现的,此外还提供了较为强大的动画能力,希望大家能支持一下下!
GitHub repo:GitHub - Xiaokang2022/tkintertools
github.io: Profile - 简介 - tkintertools (xiaokang2022.github.io)
链接文件包含了源代码,以及已经打包好,可以直接运行的 exe 文件。
源代码及打包程序下载链接:Magic Cube v1.2.zip - 蓝奏云
记得给我点赞!收藏!以及在评论区留下你的足迹呀!
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。