Faster A* For Games

Faster A* For Games Gameplay Programming @ DAE


Introduction

The technique I used is mainly called Goal Bounding, it is an optimization technique for pathfinding algorithms. In theory, it can be applied to any search graph. After some digging, I stumbled upon a paper written by Steve Rabin and Nathan R. Sturtevant [1] where they explain the core fundamentals of how this works. I'm going to take it a step further and implement it into a basic pathfinding program.

Github source
Download Executable

Main Goal

Concept

The main goal of this project is to optimize a pathfinding algorithm at runtime by pruning (excluding) large chunks of areas from a so-called "search graph". This check can be done by checking bounding boxes for each connection from a node with the optimal route to a goal node. These bounding boxes are computed at different times but before the actual runtime. Computation time will increase relative to how large your search graph is. But never the less, the algorithm at runtime will run faster because of those exclusions.

Here you can see three bounding boxes precomputed for each edge of the black node. (Source: Faster A * with goal bounding) [1]


Optimization

Requirements

The search graph must be static, in other words, it can not change. Since that would simply mean we would have to recompute all the bounding boxes. And this can not be done at runtime.

It is important if u have a large NavMesh that these precomputations are happening before release. You will have to save the data to a file, so you can allocate the memory needed for this data on the RAM when initializing the game/program. (I recommend saving the data as pure binary)


Implementation

Computation Algorithm

The approach I took is slightly different. Since my nodes are not in the centre of a triangle/square but in the middle of each edge of a triangle containing an adjacent triangle/square.

Because of this, the interpretation of terms in my code will be different from the terms mentioned in the source paper.
connection = edge = d = neighbor;
current record = node = n;

This is how the enhanced Dijkstra method works:
So what information we actually need is for each node the optimal starting connection. This information is available to us in a normal Dijkstra algorithm, it's just not stored since we don't need it in a normal Dijkstra algorithm. To actually step through each node during a Dijkstra algorithm, we don't give a Goal node (goal node is the same as start node). This gives it the effect of flood fills the whole graph. Doing this, we will have each node checked. To save this data you need to pass that connection's ID to its children, each time you push a new record to the open list.
Having this information we can compute a bounding box for each connection of a node.

Other Computation

I've stumbled upon another approach when implementing this. What if you could store for each node a container containing a container representing each node its optimal start connection.

Here you can see all the nodes containing the optimal start connection from the black node. Do that for each node and you would simply have to check if your goal node's optimal start node is the same as your currentConnection in the foreach(connection of currentNode) of the while loop. Using this in my implementation you would need to link the start and goal node each to their closest node. Because the start and goal node are not present when acquiring this information.


Runtime Algorithm

When implementing this algorithm I stumbled upon some problems. The start & end-node were not present when computing the bounding boxes.
That left me with 2 options:
1: I could find the closest valid Node and work my way up from there.
2: I could check if I'm working with a start or end node and skip the check.
I choose to work with my second option.
Check if I can do the Bounding Box check
So it was said, you need to check if the Goal Position is in a connection's assigned bounding box from the currentNode to actually continue the search. That is really it, but it was tricky to actually implement it. Since I had a dynamic Start and Goal node, which are added at run time. Since these are only 2 nodes, I did a simple check to skip them.

This is the check if the connection is related to the Start or a Goal node


Bounding Box Check

This shows how I implemented the Bounding Box check.
1: I find if the bounding Box at the node connection exists, If it doesn't exist there was no optimal route.
2: If it does exist, I check if the Goal(pos) is within that bounding box. If it is, that means that connection is the optimal start connection for that Goal node.


Conclusion

In my opinion, this can really speed up any Path Finding procedure. When considering the constraints for this method, and implementing it right. I really had fun implementing this optimization technique, it gave me a bigger perspective of what you can do with pathfinding and how large the spectrum actually is of the pathfinding branch.
If there are any questions or you want to give feedback, feel free to contact me.

Github source
Download Executable

References

Steve Rabin and Nathan R. Sturtevant, Faster A * with Goal Bounding [1]