using System.Collections.Generic; using System.Text; using UnityEngine; namespace Game { public enum NodeType { Movable, //可移動區域 Obstacle, //障礙物 Boundary, //邊界 Route //路徑 } public enum NodeState { None, //默認 Open, //開放列表 Close //封閉列表 } public class SearchPath { private readonly Dictionary<Vector2, Node> m_nodes = new Dictionary<Vector2, Node>(); private readonly List<Node> list_close = new List<Node>(); private readonly List<Node> list_open = new List<Node>(); private Vector2 position_target; private Node[,] t_nodes; /// <summary> /// 初始化地圖信息 /// </summary> /// <param name="width">寬</param> /// <param name="height">高</param> /// <param name="slant">斜向關聯</param> /// <returns>地圖數據</returns> public void InitMap(int width, int height, bool slant = false) { t_nodes = new Node[width, height]; m_nodes.Clear(); for (int i = 0; i < width; i++) { for (int j = 0; j < height; j++) { t_nodes[i, j] = new Node(i, j); if (i == 0 || i == width - 1 || j == 0 || j == height - 1) { t_nodes[i, j].type = NodeType.Boundary; } m_nodes.Add(new Vector2(i, j), t_nodes[i, j]); } } Vector2 key; //關聯周邊節點 for (int i = 0; i < width; i++) { for (int j = 0; j < height; j++) { if (slant) { for (int h = -1; h <= 1; h++) { for (int v = -1; v <= 1; v++) { if (h != 0 || v != 0) { key = new Vector2(i + h, j + v); if (m_nodes.ContainsKey(key)) { t_nodes[i, j].neighbour.Add(m_nodes[key]); } } } } } else { for (int k = -1; k <= 1; k++) { if (k != 0) { key = new Vector2(i + k, j); if (m_nodes.ContainsKey(key)) { t_nodes[i, j].neighbour.Add(m_nodes[key]); } key = new Vector2(i, j + k); if (m_nodes.ContainsKey(key)) { t_nodes[i, j].neighbour.Add(m_nodes[key]); } } } } } } } /// <summary> /// 設置障礙物 /// </summary> /// <param name="points">位置</param> public void Obstacle(params Vector2[] points) { foreach (var key in points) { if (m_nodes.ContainsKey(key)) { m_nodes[key].type = NodeType.Obstacle; } } } /// <summary> /// 尋路 /// </summary> /// <param name="nodes">地圖信息</param> /// <param name="start">起點</param> /// <param name="end">終點</param> /// <returns>路徑是否存在</returns> public bool Search(Vector2 start, Vector2 end) { bool result = false; if (!m_nodes.ContainsKey(start) || !m_nodes.ContainsKey(end)) { return result; } if (m_nodes[start].type != NodeType.Movable || m_nodes[end].type != NodeType.Movable) { return result; } //設置終點 position_target = end; //重置路徑 for (int i = 0; i < t_nodes.GetLength(0); i++) { for (int j = 0; j < t_nodes.GetLength(1); j++) { t_nodes[i, j].Reset(); } } list_close.Clear(); list_open.Clear(); Node A = t_nodes[(int)start.x, (int)start.y]; A.G = 0; A.H = Vector2.Distance(position_target, A.position); A.F = A.G + A.H; A.parent = null; A.state = NodeState.Close; list_close.Add(A); do { if (list_open.Count > 0) { A = list_open[0]; } for (int i = 0; i < list_open.Count; i++) { if (list_open[i].F < A.F) { A = list_open[i]; } } if (A.Compare(position_target)) { result = true; } Node B = Search(A); if (B != null) { do { B.type = NodeType.Route; B = B.parent; } while (B != null); } list_close.Add(A); list_open.Remove(A); A.state = NodeState.Close; } while (list_open.Count > 0); return result; } private Node Search(Node A) { Node B; for (int i = 0; i < A.neighbour.Count; i++) { if (A.neighbour[i] != null && A.neighbour[i].type == NodeType.Movable) { B = A.neighbour[i]; if (B.state == NodeState.None)//更新B的父節點爲A,並相應更新B.G; 計算B.F,B.H; B加入OpenList { B.parent = A; B.G = Vector2.Distance(A.position, B.position) + A.G; B.H = Vector2.Distance(B.position, position_target); B.F = B.G + B.H; B.state = NodeState.Open; list_open.Add(B); if (B.H < Mathf.Epsilon)//B的全部父節點既是路徑 { return B; } } else if (B.state == NodeState.Open) { float curG = Vector2.Distance(A.position, B.position); if (B.G > curG + A.G)//更新B的父節點爲A,並相應更新B.G,B.H { B.parent = A; B.G = curG + A.G; B.F = B.G + B.H; } } } } return null; } /// <summary> /// 路徑數據 /// </summary> public List<Vector2> Output { get { List<Vector2> route = new List<Vector2>(); if (m_nodes.ContainsKey(position_target)) { Node node = m_nodes[position_target]; while (node != null) { route.Add(node.position); node = node.parent; } } StringBuilder sb = new StringBuilder(); for (int i = 0; i < route.Count; i++) { sb.Append(route[i].ToString()); sb.Append("&"); } Debug.LogFormat("<color=yellow>{0}</color>", sb.ToString()); return route; } } public Node[,] GetNodes() { return t_nodes; } } public class Node { public Vector2 position; public NodeState state; public NodeType type; public float F; // F = G + H public float G; //從起點移動到指定方格的移動代價 public float H; //從指定方格移動到終點的移動代價 public Node parent; public List<Node> neighbour = new List<Node>(); public Node(int x, int y) { position = new Vector2(x, y); } public void Reset() { F = G = H = 0; parent = null; state = NodeState.None; if (type.Equals(NodeType.Route)) { type = NodeType.Movable; } } public bool Compare(Vector2 position) { return this.position.x == position.x && this.position.y == position.y; } } }
using Game; using System.Collections.Generic; using UnityEngine; public class Test : MonoBehaviour { private SearchPath path_ctr; private Dictionary<Vector2, Renderer> m_map = new Dictionary<Vector2, Renderer>(); private void Awake() { path_ctr = new SearchPath(); } private void Start() { path_ctr.InitMap(100, 100, true); for (int i = 0; i < 60; i++) { path_ctr.Obstacle(new Vector2(60, i)); } for (int i = 10; i < 60; i++) { path_ctr.Obstacle(new Vector2(i, 60)); } for (int i = 0; i < 50; i++) { path_ctr.Obstacle(new Vector2(i, 30)); } Node[,] nodes = path_ctr.GetNodes(); InitMap(nodes); Search(new Vector2(3, 3), new Vector2(80, 74)); } private void Update() { if (Input.GetKeyDown(KeyCode.P)) { Vector2 start = new Vector2(Random.Range(1, 30), Random.Range(1, 100)); Vector2 end = new Vector2(Random.Range(70, 100), Random.Range(1, 100)); Search(start, end); } } private void Search(Vector2 start, Vector2 end) { bool result = path_ctr.Search(start, end); if (result) { Debug.Log("<color=green>成功尋找到路徑!</color>" + path_ctr.Output.Count); } else { Debug.LogFormat("<color=red>未尋找到路徑,起始點:{0} 結束點{1}</color>", start, end); } Node[,] nodes = path_ctr.GetNodes(); RefreshMap(nodes); } public void InitMap(Node[,] nodes) { for (int i = 0; i < nodes.GetLength(0); i++) { for (int j = 0; j < nodes.GetLength(1); j++) { GameObject curCube = GameObject.CreatePrimitive(PrimitiveType.Cube); curCube.transform.position = new Vector3(i, j, 0); m_map.Add(new Vector2(i, j), curCube.GetComponent<Renderer>()); } } } public void RefreshMap(Node[,] nodes) { for (int i = 0; i < nodes.GetLength(0); i++) { for (int j = 0; j < nodes.GetLength(1); j++) { Vector2 key = new Vector2(i, j); if (m_map.ContainsKey(key)) { if (nodes[i, j].type == NodeType.Boundary) { m_map[key].material.SetColor("_Color", Color.black); } else if (nodes[i, j].type == NodeType.Obstacle) { m_map[key].material.SetColor("_Color", Color.red); } else if (nodes[i, j].type == NodeType.Route) { m_map[key].material.SetColor("_Color", Color.yellow); } else { m_map[key].material.SetColor("_Color", Color.white); } } } } } }
參考:https://blog.csdn.net/qq_36946274/article/details/81982691node