C#寻路,让玩家平稳地移动敌人。走动固体

时间:2015-07-18 20:36:34

标签: c# xna path-finding bounds

我需要一种方法在我的游戏中实现路径查找。

例如 A * Euclidean 对此有用。

我的 Tile 类的多维数组有一个矩形 for bounds和 is_solid 变量。玩家不仅限于瓷砖并且可以自由移动。所以敌人将顺利地找到通往玩家的路径。

我尝试在网络上或在其他 Stackoverflow 数据库上实现示例。但无济于事,我无法修改它们以适应我的项目。

有任何示例/教程您认为适合我开发游戏的方式,我可以更轻松地实现我的游戏。

1 个答案:

答案 0 :(得分:1)

我是在this great article

的帮助下这样做的
public class MyEdge//used edges and notes for my graph
{
    public int To { get; private set; }
    public int From { get; private set; }

    protected MyNode from;
    protected MyNode to;


    public MyEdge(int from, int to, MyGraph g)
    {
        From = from;
        To = to;
        this.from = g.GetNode(from);
        this.to = g.GetNode(to);
    }

    public MyNode NodeFrom()
    {
        return from;
    }

    public MyNode NodeTo()
    {
        return to;
    }
}

public class MyNode
{
    public int Index { get; private set; }
    public Vector2D Pos { get; private set; }
    public List<MyEdge> Edges { get; private set; }

    public MyNode parent = null;
    public double G = double.MaxValue;
    public double H = 0;
    public double F { get { return G + H; } }


    public MyNode(int x, int y)
    {
        Edges = new List<MyEdge>();
        Pos = new Vector2D(x, y);
        Index = -1;
    }
    public MyNode(Vector2D pos)
    {
        Edges = new List<MyEdge>();
        Pos = pos;
        Index = -1;
    }

    public MyNode(int x, int y, int idx)
    {
        Edges = new List<MyEdge>();
        Pos = new Vector2D(x, y);
        Index = idx;
    }
    public MyNode(Vector2D pos, int idx)
    {
        Edges = new List<MyEdge>();
        Pos = pos;
        Index = idx;
    }
}

注意:边有节点,节有边。这是为了获得更好的性能

public List<MyNode> GetPath(Vector2D startPoint, Vector2D endPoint)
    {
        //create the open list of nodes, initially containing only our starting node
        //create the closed list of nodes, initially empty
        List<MyNode> open = new List<MyNode>();
        List<MyNode> closed = new List<MyNode>();


        //Find the starting node
        MyNode start = FindNearestNode(startPoint);
        if (start == null)
            return null;

        //Find the ending node
        MyNode end = FindNearestNode(endPoint);
        if (end == null)
            return null;

        start.G = G(start.Pos, endPoint);
        start.H = H(start.Pos, endPoint);
        open.Add(start);


        while (open.Count != 0)
        {
            //consider the best node in the open list (the node with the lowest f value)
            double d = open.Min(x => x.F);
            MyNode curr = open.First(x => x.F.CompareTo(d) == 0);

            closed.Add(curr);
            open.Remove(curr);

            //If current node is goal              
            if (curr == end)
            {
                //Loop over the parents to get the path
                List<MyNode> ns = new List<MyNode>();
                MyNode n = end;
                ns.Add(n);
                while (n != start)
                {
                    ns.Add(n.parent);
                    n = n.parent;
                }
                ns.Reverse();
                return ns;
            }

            for (int i = 0; i < curr.Edges.Count; i++)
            {
                MyNode possibleNeigbour = curr.Edges[i].NodeTo();
                if (closed.Contains(possibleNeigbour))
                {
                    //ignore it
                }
                else if (!open.Contains(possibleNeigbour))
                {
                    possibleNeigbour.G = G(possibleNeigbour.Pos, endPoint);
                    possibleNeigbour.H = H(possibleNeigbour.Pos, endPoint);
                    possibleNeigbour.parent = curr;
                    open.Add(possibleNeigbour);
                }
                else if(open.Contains(possibleNeigbour))
                {
                    if (possibleNeigbour.G < curr.G)
                    {
                        possibleNeigbour.parent = curr;
                    }
                }
            }
        }

        return null;
    }