了解一个流行人工智能搜索算法的Java实现

通过搜寻可行解决方案空间来">解决问题是人工智能中一项名为状态空间搜索 的基本技术。 启发式搜索 是状态空间搜索的一种形式,利用有关一个问题的知识来更高效地查找解决方案。启发式搜索在各个领域荣获众多殊荣。在本文中,我们将向您介绍启发式搜索领域,并展示如何利用 Java 编程语言实现 A*,即最广为使用的启发式搜索算法。启发式搜索算法对计算资源和内存提出了较高的要求。我们还将展示如何避免昂贵的垃圾收集,以及如何利用一个替代的高性能 Java 集合框架 (JC++F),通过这些改进 Java 实现。

启发式搜索

计算机科学中的许多问题可用一个图形数据结构表示,其中图形中的路径表示潜在的解决方案。查找最优解决方案需要找到一个最短路径。例如,以自主视频游戏角色为例。角色做出的每个动作都与图形中的一个边缘相对应,而且角色的目标是找到最短路径,与对手角色交手。

深度优先 搜索和广度优先 搜索等算法是流行的图形遍历算法。但它们被视为非启发式 算法,而且常常受到它们可以解决的问题规模的严格限制。此外,不能保证深度优先搜索能找到最优解决方案(或某些情况下的任何解决方案),可以保证广度优先搜索仅能在特殊情况下找到最优解决方案。相比之下,启发式搜索是一种提示性 搜索,利用有关一个问题的知识,以启发式 方式进行编码,从而更高效地解决问题。启发式搜索可以解决非启发式算法无法解决的很多难题。

视频游戏寻路是启发式搜索的一个受欢迎的领域,它还可以解决更复杂的问题。2007 年举行的无人驾驶汽车比赛 “DARPA 城市挑战赛” 的优胜者就利用了启发式搜索来规划平坦的、直接的可行使路线。启发式搜索在自然语言处理中也有成功应用,它被用于语音识别中的文本和堆栈解码句法解析。它在机器人学和生物信息学领域也有应用。与传统的动态编程方法相比较,使用启发式搜索可以使用更少的内存更快地解决多序列比对 (Multiple Sequence Alignment, MSA),这是一个经过深入研究的信息学问题。

通过 Java 实现启发式搜索

Java 编程语言不是实现启发式搜索的一种受欢迎的选择,因为它对内存和计算资源的要求很高。出于性能原因,C/C++ 通常是首选语言。我们将证明 Java 是实现启发式搜索的一种合适的编程语言。我们首先表明,在解决受欢迎的基准问题集时,A* 的 textbook 实现确实很缓慢,并且会耗尽可用内存。我们通过重访一些关键实现细节和利用替代的 JCF 来解决这些性能问题。

很多这方面的工作都是本文作者合著的一篇学术论文中发表的作品的一个扩展。尽管原作专注于 C/C++  编程,但在这里,我们展示了适用于 Java 的许多同样的概念。

广度优先搜索

熟悉广度优先搜索(一个共享许多相同概念和术语的更简单的算法)的实现,将帮助您理解实现启发式搜索的细节。我们将使用广度优先搜索的一个以代理为中心 的视图。在一个以代理为中心的视图中,代理据说处于某种状态,并且可从该状态获取一组适用的操作。应用操作可将代理从其当前状态转换到一个新的后继 状态。该视图适用于多种类型的问题。

广度优先搜索的目标是设计一系列操作,将代理从其初始状态引导至一个目标状态。从初始状态开始,广度优先搜索首先访问最近生成的状态。所有适用的操作在每个访问状态都可以得到应用,生成新的状态,然后该状态被添加到未访问状态列表(也称为搜索的前沿)。访问状态并生成所有后继状态的过程被称为扩展 该状态。

您可以将该搜索过程看作是生成了一个树:树的根节点表示初始状态,子节点由边缘连接,该边缘表示用于生成它们的操作。图 1 显示该搜索树的一个图解。白圈表示搜索前沿的节点。灰圈表示已展开的节点。

图 1. 二叉树上的广度优先搜索顺序

搜索树中的每一个节点表示某种状态,但两个独特的节点可表示同一状态。例如,搜索树中处于不同深度的一个节点可以与树中较高层的另一个节点具有同样的状态。这些重复 节点表示在搜索问题中达到同一状态的两种不同方式。重复节点可能存在问题,因此必须记住所有受访节点。

清单 1 显示广度优先搜索的伪代码:

清单 1. 广度优先搜索的伪代码

function: BREADTH-FIRST-SEARCH(initial)open ← {initial}closed ← 0loop do: if EMPTY(open) then return failure node ← SHALLOWEST(open) closed ← ADD(closed, node) for each action in ACTIONS(node) successor ← APPLY(action, node) if successor in closed then continue if GOAL(successor) then return SOLUTION(node) open ← INSERT(open, successor)

在 清单 1 中,我们将搜索前沿保留在一个 open 列表(第 2 行)中。将访问过的节点保留在 closed 列表(第 3 行)中。closed 列表有助于确保我们不会多次重访任何节点,从而不会重复搜索工作。仅当一个节点不在 closed 列表中时才能将其添加到前沿。搜索循环持续至 open 列表为空或找到目标为止。

在 图 1 中,您可能已经注意到,在移至下一层之前,广度优先搜索会访问搜索树的每个深度层的所有节点。在所有操作具有相同成本的问题中,搜素树中的所有边缘具有相同的权重,这样可保证广度优先搜索能找到最优解决方案。也就是说,生成的第一个目标在从初始状态开始的最短路径上。

在某些域中,每个操作有不同的成本。对于这些域,搜索树中的边缘具有不统一的权重。在这种情况下,一个解决方案的成本是从根到目标的路径上所有边缘权重的总和。对于这些域,无法保证广度优先搜索能找到最优解决方案。此外,广度优先搜索必须展开树的每个深度层的所有节点,直至生成目标。存储这些深度层所需的内存可能会快速超过最现代的计算机上的可用内存。这将广度优先搜索限制于很窄的几个小问题。

Dijkstra 的算法是广度优先搜索的一个扩展,它根据从初始状态到达节点的成本对搜索前沿上的节点进行排序(排列 open 列表)。不管操作成本是否统一(假设成本是非负值),它都确保可以找到最优解决方案。然而,它必须访问成本少于最优解决方案的所有节点,因此它被限制于解决较少的问题。下一节将描述一个能解决大量问题的算法,该算法能大幅减少查找最优解决方案所需访问的节点数量。

时间: 2024-08-30 05:52:37

了解一个流行人工智能搜索算法的Java实现的相关文章

android-一个安卓工程引入无法另一个安卓工程的R.java

问题描述 一个安卓工程引入无法另一个安卓工程的R.java 我有2个安卓工程,一个工程作为library导入到另外一个工程,其他类的引用正常,但作为library的工程的R.java无法引入. 解决方案 如果libary工程下的R.java能够正常自动生成的话,你引用libary的R使用完整的路径试试,如xxx.library.R.drawable.xxx

在做一个游戏,想问java有没有类似图层功能

问题描述 在做一个游戏,想问java有没有类似图层功能 目前已知情报 1.好像有个叫JlayerdPane的 2.我的目标是顶层游戏主体用panel话,中层添加视频,底层添加图片 3.问题有两个: 一是用media只能放在frame上面?我想加在panel上,否则添加panel的时候就被覆盖掉了 二是把panel设置成背景透明的,通过 panel.setBackground(null); // 把背景设置为会 panel.setOpaque(false); // 设置为透明 不知道在一的条件下能

前台是个图标,<img>图标。 怎么调用JSP作成的日历页面(一个小页面,用java + html代码,不是js)

问题描述 前台是个图标,<img>图标.怎么调用JSP作成的日历页面(一个小页面,用java+html代码,不是js)怎么调用这个jsp日历呢??另外在这个JSP里面输入的值,是怎么传回呢?? 解决方案 解决方案二:我也等回答.....解决方案三:该回复于2010-12-03 10:18:23被版主删除解决方案四:吧图片做成个链接<ahref="***.jsp"><imgsrc=""></a>解决方案五:可以将你的be

java 服务器-各位业界大神,怎么实现一个服务器挂多个java网站。

问题描述 各位业界大神,怎么实现一个服务器挂多个java网站. 大家都知道一个服务器挂多个java网站有些困难,除了修改tomcat的server文件的host,可以通过这个方法挂上多java网站,但是网站都不稳定,访问量大了.tomcat自己都崩溃了,所以网站都运不了,请大神,给一个非常好的方法可以在同一个服务器上挂多个网站的更好的方法. 解决方案 挂载到云服务器上试一下!!!

java中文乱码-java开发前端ftl页面上传一个中文的字段到java后台变成乱码!

问题描述 java开发前端ftl页面上传一个中文的字段到java后台变成乱码! 有没有比在后台写一些转换格式的代码更简单的方式?是不是我哪里的格式没有设置好? 解决方案 找到答案了.原来是我tomcat的编码格式没有设置成utf-8. 解决方案二: ftl文件默认的编码格式是GBK的,但是它输出的编码格式是utf-8的.中文就会显示乱码.在改变ftl文件的编码格式之前,将页面整个copy到txt文档中,然后改变编码格式之后,再将txt文件中的内容复制到原来的文件中. 解决方案三: 客户端 编码格

单链表-寻找一个单向链表的中项 java 调用问题!谢谢大家

问题描述 寻找一个单向链表的中项 java 调用问题!谢谢大家 //寻找一个单向链表的中项,如果存在两个则返回前一个,给出算法描述,并实现 package pra_bd; import java.awt.DisplayMode; import javax.naming.spi.DirStateFactory.Result; import org.w3c.dom.Node; public class LinkList { //头结点 Link first; //单链表构造函数 public Lin

求一个抽奖算法,最好java实现

问题描述 求一个抽奖算法,最好java实现 需求如下: 总共分四等奖, 1等奖名额2个,中奖概率2/10000 2等奖名额10个,中奖概率5/1000 3等奖名额100个,中奖概率10/100 4等奖名额880个,中奖概率25/100 求一个算法,算出用户单次抽奖中奖等级(1,2,3,3等奖或不中奖). 最好java实现. 解决方案 说个最简单的,用java的随机函数.用Random类生成随机数,也可以用Math类中的random生成 区别看这里:http://liukai.iteye.com/

请问一个广州地区的手机JAVA客户端软件开发人员月薪大概多少?

问题描述 请问一个广州地区的手机JAVA客户端软件开发人员月薪大概多少?有人知道吗? 解决方案 解决方案二:该回复于2009-08-28 13:37:37被版主删除

一个项目反编译成java文件后怎样编译成class文件

问题描述 本人是java新手,一个项目要修改一些数据,用jd-gui把class文件全部反编译成java文件,请问怎样重新编译成class文件. 解决方案 解决方案二:javacfile.java命令可以编译file.java为file.classjavafile可以运行file.class,注意命令行中不用后缀解决方案三:一个项目不是单个java解决方案四:一个项目的话,直接在eclipse直接运行时,应该自动编译了吧解决方案五:弄进Eclipse,就完事了...解决方案六:引用1楼voice