A星尋路

邏輯代碼
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;
        }
    }
}
View Code
調用入口 Test
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);
                    }
                }
            }
        }
    }
}
View Code

 

 

參考:https://blog.csdn.net/qq_36946274/article/details/81982691node

相關文章
相關標籤/搜索