Interactive visualization of maze generation and graph traversal algorithms in JavaScript and HTML5 Canvas. The maze is generated using randomized Prim's algorithm to build a minimum spanning tree. The tree can be searched with several graph traversal algorithms.
This demonstration considers the entire grid as a weighted, undirected graph for pathfinding. It is undirected since each node is assigned a collection of all of its neighboring nodes and edges available for travel. It is weighted since each edge has a random float weight between 0 and 1 (exclusive). The maze is a spanning tree where both nodes and edges are drawn to the canvas.
Prim's algorithm first chooses a source node, observes the node's neighbors and compares each edge's weight. Then, it chooses the passage with the local minimum edge weight as its next path and walks to the next node. As it does this, it constructs a spanning tree of all the locally optimal nodes and edges it reaches. The result is a subgraph of the original fully connected graph. This minimum spanning tree has the lowest possible total edge weight. Since it's only concerned with local minima edge weights, Prim's is a greedy algorithm.
In randomized Prim's algorithm, the maze's edges are initialized with random edge weights. The maze is built in the direction of the local minimum edge weight (cheapest edge) at each node if and only if the extension would avoid creating a loop in the maze. In the case that Prim's compares edge weights and predicts a cycle, the next available local minimum edge weight is chosen.
To keep track of the cells (graph nodes) that can potentially be incorporated into the minimum spanning tree, a frontier of cells that wall in the current position in the tree is maintained. This frontier is repeatedly explored, evaluated, and adjusted until the spanning tree contains all of the nodes.
First, a fully-connected graph is generated. Each node is assigned neighboring edges in each available direction. The Edge
constructor initializes each edge with a random weight.
Generation arbitrarily starts from the top-left corner (canvas coordinates 0, 0). The algorithm adjusts a frontier of available cells, explores it for the lowest-weighted edge, draws the current node and chosen edge, and finally extends the minimum spanning tree. When the algorithm predicts a cycle, it backtracks to the next available node based on the frontier. This is made possible by keeping the discovered
state of each node (in this implementation, it is stored on each Node
instance). The loop is repeated until the spanning tree contains all of the nodes in the initial graph.
The colors are based on the progress of maze generation. Lighter colors indicate the nodes and edges were placed earlier, and darker colors indicate late placement. Since it's very common for Prim's to run into cycles after most of the maze has been generated, you can expect to see how backtracking returns to random spots in the maze.
Breadth-first search visits every node's incident edges before descending to the next level of nodes (siblings first).
It is guaranteed to find the shortest path on an undirected graph. Visiting every increasingly distant node from the source eventually generates a unique path to the target with the smallest number of edges.
Breadth-first search is implemented with a queue, which maintains the order for traversal. Dequeueing returns the earliest added node, allowing the search to continue exploring a particular level of the graph before moving on and considering nodes at lower levels.
Depth-first search also considers the neighbors of each node. After walking to a node, the search keeps traveling to the deepest level before backtracking (children first: visit the parent node's children first).
It's implemented with a stack. Iterative implementations make use of an explicit stack, and recursive implementations use an implicit stack.
Depth-first search is not guaranteed to find the shortest-path from the source to the target - it might terminate with a suboptimal solution.
In the worst case, both search algorithms visit each vertex at most once and cover each edge at least once. They have the same worst-case time (O(|V| + |E|)
) and space complexity (O(|V|)
). BFS is guaranteed to find the shortest-path solution, but might be slower to reach the solution when the target node is far from the source node. DFS can potentially be more useful when the target node is far from the source node, but it typically finds suboptimal solutions and might also traverse the entire graph.
Both BFS and DFS have simple iterative implementations. BFS makes use of a queue while DFS makes use of a stack. JavaScript's asynchronous setInterval()
is used to repeatedly draw traversal and generation to the canvas. setInterval()
returns an ID value that can be fed to clearInterval()
to cancel any building paths.
- Visualize the graph during generation and traversal using Sigma.js
- Implement a min-heap for randomized Prim's algorithm
- Implement hexagonal maze generation (introduction to hexagonal grids)
- Uniform-cost search and Dijkstra's algorithm
- Red Blob Games
- Jamis Buck
- Introduction to Maze Generation Algos
- His full book Mazes for Programmers (examples in Ruby)
- Steven Skiena's Lectures:
- Mike Bostock's work on Prim's Algorithm vs. random traversal
- Daniel Shiffman - The Coding Train
- Algorithms
- Prim's
- Priority Queues
- A* Search