我需要一种方法在我的游戏中实现路径查找。
例如 A * 或 Euclidean 对此有用。
我的 Tile 类的多维数组有一个矩形 for bounds和 is_solid 变量。玩家不仅限于瓷砖并且可以自由移动。所以敌人将顺利地找到通往玩家的路径。
我尝试在网络上或在其他 Stackoverflow 数据库上实现示例。但无济于事,我无法修改它们以适应我的项目。
有任何示例/教程您认为适合我开发游戏的方式,我可以更轻松地实现我的游戏。
答案 0 :(得分:1)
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;
}