package astar; import java.util.ArrayList; import java.util.Comparator; import java.util.List; import java.util.PriorityQueue; /** * A*搜索算法,A星算法。 * 这是一种在图形平面上,有多个节点的路径,求出最低通过成本的算法。 * 常用于游戏中的NPC的移动计算,或在线游戏的BOT的移动计算上。 * 该算法像Dijkstra算法一样,可以找到一条最短路径;也像BFS一样,进行启发式的搜索。 * 在此算法中,如果以 g(n)表示从起点到任意顶点n的实际距离, * h(n)表示任意顶点n到目标顶点的估算距离, * 那么 A*算法的公式为:f(n)=g(n)+h(n)。 这个公式遵循以下特性: * 如果h(n)为0,只需求出g(n),即求出起点到任意顶点n的最短路径,则转化为单源最短路径问题,即Dijkstra算法 * 如果h(n)<=“n到目标的实际距离”,则一定可以求出最优解。而且h(n)越小,需要计算的节点越多,算法效率越低。 * * @author DBJ(dubenju@126.com) */ public class AStar { /** 搜索区域 0:不能走 1:能走 */ private int[][] searchArea; /** 记录节点状态 */ private int[][] searchAreaStatus; private int width; private int height; /** 开启列表 */ private PriorityQueue<AStarNode> openList; /** FComparator */ public static Comparator<AStarNode> fComparator = new Comparator<AStarNode>() { @Override public int compare(AStarNode c1, AStarNode c2) { return (int) (c1.getF() - c2.getF()); } }; /** * AStart * @param searchArea 搜索区域 * @param width 搜索区域的宽 * @param height 搜索区域的高 */ public AStar(int[][] searchArea, int width, int height) { this.width = width; this.height = height; this.searchArea = searchArea; this.searchAreaStatus = new int[this.height][this.width]; this.openList = new PriorityQueue<>(10, fComparator); } /** * 查找路径 * * @param x1 起点A的x坐标 * @param y1 起点A的y坐标 * @param x2 终点B的x坐标 * @param y2 终点B的y坐标 * @return 起点A到终点B的路径 */ public List<AStarNode> find(int x1, int y1, int x2, int y2) { AStarNode startNode = new AStarNode(x1, y1); AStarNode endNode = new AStarNode(x2, y2); return this.find(startNode, endNode); } /** * find 搜索 * @param startNode 起点 * @param endNode 终点 * @return 路径 */ private List<AStarNode> find(AStarNode startNode, AStarNode endNode) { List<AStarNode> resultList = new ArrayList<AStarNode>(); /* 是否找到 */ boolean findFalg = false; /* 1.从起点A开始,并将它添加到 “开启列表”。 */ this.openList.add(startNode); searchAreaStatus[startNode.getX()][startNode.getY()] = AStarConstants.NOTE_STATUS_OPEN; AStarNode curNode = this.openList.poll(); while (curNode != null) { /* c)对于当前方格临近的8个方格的每一个....(For Each) */ // System.out.println("find@AStar curNode=" + curNode); for (int i = 0; i < 8; i++) { switch (i) { case 0:// 右 check(curNode.getX(), curNode.getY() + 1, endNode, curNode, AStarConstants.COST_ORTHOGONAL); // 10 break; case 1:// 下 check(curNode.getX() + 1, curNode.getY(), endNode, curNode, AStarConstants.COST_ORTHOGONAL); // 10 break; case 2:// 左 check(curNode.getX(), curNode.getY() - 1, endNode, curNode, AStarConstants.COST_ORTHOGONAL); // 10 break; case 3:// 上 check(curNode.getX() - 1, curNode.getY(), endNode, curNode, AStarConstants.COST_ORTHOGONAL); // 10 break; case 4:// 右上 check(curNode.getX() - 1, curNode.getY() + 1, endNode, curNode, AStarConstants.COST_DIAGONAL); // 14 break; case 5:// 右下 check(curNode.getX() + 1, curNode.getY() + 1, endNode, curNode, AStarConstants.COST_DIAGONAL); // 14 break; case 6:// 左上 check(curNode.getX() - 1, curNode.getY() - 1, endNode, curNode, AStarConstants.COST_DIAGONAL); // 14 break; case 7:// 左下 check(curNode.getX() + 1, curNode.getY() - 1, endNode, curNode, AStarConstants.COST_DIAGONAL); // 14 break; } // end switch } // end for // 加入关闭列表 findFalg = this.addClosedList(curNode, endNode); if (findFalg) { break; } /* a)寻找开启列表上最小F值的方格。将它作为当前方格。 */ curNode = this.openList.poll(); } // while if (findFalg) { // 有 read(resultList, curNode); return resultList; } else { /* 无法找到目标方格并且开启列表是空的时候,不存在路径。 */ return resultList; } } /** * 读取所有节点,从起点开始返 * * @param resultList * @param node */ private void read(List<AStarNode> resultList, AStarNode node) { if (node.getParent() != null) { read(resultList, node.getParent()); } resultList.add(node); } /** * hasNearbyUnwalkable * @param x x坐标 * @param y y坐标 * @param parent 父 * @return */ private boolean hasNearbyUnwalkable(int x, int y, AStarNode parent) { boolean bRes = false; if (x != parent.getX() && y != parent.getY()) { if (this.searchArea[parent.getX()][y] == AStarConstants.NOTE_UNWALKABLE) { bRes = true; } if (this.searchArea[x][parent.getY()] == AStarConstants.NOTE_UNWALKABLE) { bRes = true; } } return bRes; } /** * 检查 当前节点周围的节点,是否能行,是否在开启列表中,是否在关闭列表中 如果不在关闭与开启列表中则加入开启列表,如果在开启中则更新节点G值信息 * * @param x x坐标 * @param y y坐标 * @param endNode 终点 * @param parent 父 * @param step 步长 * @return */ private boolean check(int x, int y, AStarNode endNode, AStarNode parent, int step) { // System.out.println(" check@AStar (" + x + "," + y + ")" + parentNode); try { if (this.searchArea[x][y] == AStarConstants.NOTE_UNWALKABLE) { /* 如果不能走,忽略它。*/ return false; } if (this.searchAreaStatus[x][y] == AStarConstants.NOTE_STATUS_CLOSED) { /* 如果它在关闭列表上,忽略它。 */ return false; } /* 否则,执行以下操作。 */ if (this.searchAreaStatus[x][y] == AStarConstants.NOTE_STATUS_NONE) { if (hasNearbyUnwalkable(x, y, parent)) { return false; } /* 如果不在开启列表中,把它添加到开启列表。使当前方格成为这个方格的父。记录的方格F值,G值和H值。*/ this.addOpenList(x, y, endNode, parent, step); this.searchAreaStatus[x][y] = AStarConstants.NOTE_STATUS_OPEN; return true; } else if (this.searchAreaStatus[x][y] == AStarConstants.NOTE_STATUS_OPEN) { /* 如果在开启列表了,检查看看采用G值来衡量这个路径到那个方格是否是更好的。*/ this.updateOpenList(x, y, endNode, parent, step); return true; } } catch (ArrayIndexOutOfBoundsException e) { return false;// 下标越界 } return false; } /** * 添加到关闭列表 * * @param node 节点 * @param endNode 终点 * @return true:路径被发现 */ private boolean addClosedList(AStarNode node, AStarNode endNode) { if (node.getX() == endNode.getX() && node.getY() == endNode.getY()) { /* 在目标方格添加到关闭列表的情况下,路径已经被发现 */ return true; } this.searchAreaStatus[node.getX()][node.getY()] = AStarConstants.NOTE_STATUS_CLOSED; return false; } /** * 添加到开启列表 * 使当前方格成为这个方格的父。记录的方格F值,G值和H值。 * * @param x x坐标 * @param y y坐标 * @param endNode 终点 * @param parent 父 * @param step 步长 * @return */ private boolean addOpenList(int x, int y, AStarNode endNode, AStarNode parent, int step) { /* 使当前方格成为这个方格的父。 */ AStarNode node = new AStarNode(x, y, parent); /* 记录的方格F值,G值和H值。 */ this.count(node, endNode, step); this.openList.add(node); // System.out.println(" putOpenTable@AStar " + node + parentNode); return true; } /** * 已经在开启列表了更新节点F值 * 更低的G值意味着这是一个更好的路径。如果是这样,把方格的父改为当前方格,并重新计算方格的G值和F值。如果你保持开启列表排序F值,由于这个变化你可能需重存列表。 * * @param x x坐标 * @param y y坐标 * @param endNode 终点 * @param parent 父 * @param step 步长 * @return */ private boolean updateOpenList(int x, int y, AStarNode endNode, AStarNode parent, int step) { AStarNode node = new AStarNode(x, y); for (AStarNode nd : this.openList) { if (node.equals(nd)) { node = nd; break ; } } int g = parent.getG() + step; if (g < node.getG()) { /* 如果是更低的G值意味着这是一个更好的路径。把方格的父改为当前方格 */ node.setParent(parent); /* 并重新计算方格的G值和F值。*/ this.count(node, endNode, step); /* 如果你保持开启列表按F值排序,由于这个变化你可能需重存列表。 */ this.openList.remove(node); this.openList.add(node); return true; } return false; } /** * 计算GHF * * @param node 节点 * @param endNode 终点 * @param step 步长 */ private void count(AStarNode node, AStarNode endNode, int step) { this.countG(node, node.getParent(), step); this.countH(node, endNode); this.countF(node); } /** * 计算G值 * 将指定每个移动水平或垂直方成本为10,对角线移动成本为14 * 找出那个方块的父亲的G值,然后加10或14取决于它从父方格是正交(非对角线)还是对角线。 * * @param node 节点 * @param parent 父 * @param step 步长 */ private void countG(AStarNode node, AStarNode parent, int step) { if (parent == null) { node.setG(step); } else { node.setG(parent.getG() + step); } } /** * 计算H值 * 曼哈顿方法 * 计算从当前方块到目标方水平和垂直方向移动方块的总数 * 将总数乘以10 * * @param node 节点 * @param endNode 终点 */ private void countH(AStarNode node, AStarNode endNode) { node.setH((Math.abs(endNode.getX() - node.getX()) + Math.abs(endNode.getY() - node.getY())) * 10); } /** * 计算F值 * F = G + H * * @param node 节点 */ private void countF(AStarNode node) { node.setF(node.getG() + node.getH()); } }