This is a snippet of the "Graphs" chapter of the book.
Graphs have already been mentioned several times, in the book, in quite diverse contexts. Actually, if you are familiar with graphs you can spot opportunities to use them in quite different areas for problems that aren't explicitly formulated with graphs in mind. So, in this chapter, we'll discuss how to handle graphs to develop such intuition to some degree.
But first, let's list the most prominent examples of the direct graph applications, some of which we'll see here in action:
- pathfinding
- network analysis
- dependency analysis in planning, compilers, etc.
- various optimization problems
- distributing and optimizing computations
- knowledge representation and reasoning with it
- meaning representation in natural language processing
Graphs may be thought of as a generalization of trees: indeed, trees are, as we said earlier, connected directed acyclic graphs. But there's an important distinction in the patterns of the usage of graphs and trees. Graphs, much more frequently than trees, have weights associated with the edges, which adds a whole new dimension both to algorithms for processing them and to possible data that can be represented in the graph form. So, while the main application of trees is reflecting some hierarchy, for graphs, it is often more about determining connectedness and its magnitude, based on the weights.
Graph Representations
A graph is, basically, a set of nodes (called "vertices", V
) and an enumeration of their pairs ("edges", E
). The edges may be directed or undirected (i.e. bidirectional), and also weighted or unweighted. There are many ways that may be used to represent these sets, which have varied utility for different situations. Here are the most common ones:
- as a linked structure:
(defstruct node data links)
where links
may be either a list of other node
s, possibly, paired with weights, or a list of edge
structures represented as (defsturct edge source destination weight)
. For directed graphs, this representation will be similar to a singly-linked list but for undirected — to a heavier doubly-linked one
- as an adjacency matrix (
V x V
). This matrix is indexed by vertices and has zeroes when there's no connection between them and some nonzero number for the weight (1 — in case of unweighted graphs) when there is a connection. Undirected graphs have a symmetric adjacency matrix and so need to store only the abovediagonal half of it
- as an adjacency list that enumerates for each vertex the other vertices it's connected to and the weights of connections
- as an incidence matrix (
V x E
). This matrix is similar to the previous representation, but with much more wasted space. The adjacency list may be thought of as a sparse representation of the incidence matrix. The matrix representation may be more useful for hypergraphs (that have more than 2 vertices for each edge), though
- just as a list of edges
More details about of the book may be found on its website.