Return to the Lecture Notes Index

15-111 Lecture 18 (Friday, June 13, 2003)

Graphs

Suppose you need to drive from Pittsburgh to Charleston. There are probably many ways to get there using different small roads, but an easy way is to follow I-79 south. Similarly, if we wanted to go from Pittsburgh to Philadelphia, one simple way is to use the PA Turnpike. Pittsburgh is linked to Charleston by I-79, and it is linked to Philadelphia by the turnpike.

Suppose now that you need to go from Philadelphia to Charleston. Without any knowledge of other roads, you know of at least one way to do that: take the turnpike from Philadelphia to Pittsburgh, and then take I-79 from Pittsburgh to Charleston. This might be an inefficient route to take, but it will get you from Philadelphia to Charleston.

A graph is a data structure which allows us to model these kinds of relationships. A graph is a collection of vertices (such as major cities) and edges (roads between those major cities). Any collection of vertices and edges is a graph. For example, the following is a graph:

This graph does not demonstrate any noticeable pattern, but from certain points we can get to other points. Some graphs do have a recognizable pattern such as:

The first graph is what we call a complete or fully-connected graph. In a fully-connected graph, every vertex has an edge to every other vertex.

The graph on the right should seem familiar to you. If we bunch up all of the edges at the bottom so that the center is now on top, this should look strikingly similar to a generic tree. This is not a coincidence -- trees are graphs. In fact, they are a very specific type of graph: one without any cycles. A cycle is a sequence of edges that creates a path back to the vertex it starts at without reusing any of the edges.

We used cities and roads as an example of what a graph might represent. Another common use for graphs is to model a maze.

In the case of a maze, the vertices represent each spot which you could ever be at, and the edges join those spots to all of the spots you could move to from there. As we will see, how you choose which spot to go to next could affect what path through the maze you eventually find.

Weighted Graphs

What if we wanted to go from Philadelphia to Charleston? We could go through Pittsburgh, but what if there is a better way? If we start out in a more southwestern direction, we might be able to find a faster route from Philadelphia to Charleston. On the other hand, while there might be a more direct path, the roads involved might be small slow roads, so the time that it takes might actually be longer even though the path is shorter.

In a graph, we can factor these details in when we are trying to find the best way to get from one place to another. Rather than simply indicating that there is an edge between two vertices, we will also give that edge a weight. If we are trying to find the fastest path from Philadelphia to Charleston, then the weight we use might be the expected time it takes on each part of the trip. If we are trying to find the shortest path, the weight we use might be the actual length of each part of the trip. We could also use some combination of these values.

We call this a weighted graph. For convenience, we will say that the weight will be infinity if there is no edge between two vertices, and that there will be a weight of 0 for staying at the current node.

Directed Graphs

Again returning to the road analogy, there is the concept of a one-way street. If you are at the Cathedral Of Learning and need to go to the highway, you have to use Fifth Ave., but if you are coming from the highway to the Cathedral Of Learning, you need to use Forbes Ave. instead, because both streets are one-way.

We might also want to have this type of behavior in our graph. We might want to be able to go from vertex A to vertex B directly, but not be able to go from vertex B to vertex A. We call this a directed graph. In a directed graph we say than an edge goes "from" one vertex "to" another vertex.

A directed graph can also be weighted. It might be really easy to go directly from vertex A to vertex B, but it might be very difficult (though possible) to go directly from vertex B to vertex A. In this case, there would be an edge from A to B and an edge from B to A, but the edge from B to A would have a much higher weight than the edge from A to B.

How Do We Represent Graphs?

Now that we understand what a graph is, how do we represent them on a computer. With the Linked List and the Binary Tree it was easy, but graphs are unstructured. One vertex might be adjecent to every other vertex, but another vertex might only be adjacent to just that one. (By adjacent, we mean that there is an edge between them, or that there is an edge from that one to the other in the case of a directed graph.)

There are two common ways of representing graphs. One is called an adjacency list, the other is called an adjacency matrix.

Adjacency List

An adjacency list representation is essentially an array of linked lists, one for each vertex, where the linked list contains all of the vertices that are adjacent to a given vertex, and in the case of a weighted graph, the weight between that vertex and each of the others. Suppose we have the following graph (the naming of the vertices is arbitrary):

Then our adjacency list would look like:

Why is this? Well, vertex 0 only has an edge with vertex 2, but vertex 2 has edges with vertex 0, vertex 1, vertex 3, and vertex 4, so the only item in the list at index 0 is 2, while at index 2 the list contains 0, 1, 3, and 4. If this were a weighted graph, the nodes of the list would need to include both the number of the vertex and the cost to get there. If this were a directed graph, then an edge from 0 to 2 would not necessarily mean that there would be an edge from 2 to 0.

The following code implements an adjacency list which can be used for graphs that are either weighted or unweighted, and either directed or bidirected (the edges go in both directions).

class AdjList
{
	/*
	 * this class represents a weighted edge.  if the graph
	 * is unweighted, we will assign the weight of an existing
	 * edge to be 0
	 */
	private class Edge
	{
		private int vertex;
		private int cost;

		public Edge(int vertex, int cost)
		{
			this.vertex = vertex;
			this.cost = cost;
		}

		public Edge(int vertex)
		{
			this.vertex = vertex;
			this.cost = 0;
		}

		/*
		 * considers two edges equal if they go to the same
		 * vertex, regardless of their weight
		 */
		public boolean equals(Edge other_edge)
		{
			if (other_edge.vertex == this.vertex)
			{
				return true;
			}
			else
			{
				return false;
			}
		}
	}

	private LinkedList list[];
	private int num_vertices;

	public AdjList(int num_vertices)
	{
		this.num_vertices = num_vertices;
		list = new LinkedList[num_vertices];

		/*
		 * initialize all of the linked lists in the array
		 */
		for (int index = 0; index < num_vertices; index++)
		{
			list[index] = new LinkedList();
		}
	}

	/*
	 * add a directed weighted edge to the graph
	 */
	public addEdge(int vertex_from, int vertex_to, int cost)
	{
		list[vertex_from].addHead(new Edge(vertex_to, cost));
	}

	/*
	 * add a directed unweighted edge to the graph
	 */
	public addEdge(int vertex_from, int vertex_to)
	{
		addEdge(vertex_from, vertex_to, 0);
	}

	/*
	 * add a bidirected weighted edge to the graph
	 *
	 * to do this, we will add a directed edge in both directions
	 * with the same weight
	 */
	public addBidirectedEdge(int vertex_one, int vertex_two, int cost)
	{
		addEdge(vertex_one, vertex_two, cost);
		addEdge(vertex_two, vertex_one, cost);
	}

	/*
	 * add a bidirected unweighted edge to the graph
	 *
	 * to do this, we will add a directed edge in both directions
	 */
	public addBidirectedEdge(int vertex_one, int vertex_two)
	{
		addEdge(vertex_one, vertex_two, 0);
		addEdge(vertex_two, vertex_one, 0);
	}

	/*
	 * remove a directed edge from the graph
	 */
	public removeEdge(int vertex_from, int vertex_to)
	{
		list[vertex_from].remove(new Edge(vertex_to));
	}

	/*
	 * remove a bidirected edge from the graph
	 *
	 * to this, we will remove a directed edge in both directions
	 */
	public removeBidirectedEdge(int vertex_one, int vertex_two)
	{
		removeEdge(vertex_one, vertex_two);
		removeEdge(vertex_two, vertex_one);
	}

	/*
	 * return back the cost of a given edge
	 */
	public int getEdge(int vertex_from, int vertex_to)
	{
		Edge foundEdge = list[vertex_from].find(new Edge(vertex_to));

		if (null == foundEdge)
		{
			return Integer.MAX_VALUE;
		}
		else
		{
			return foundEdge.cost;
		}
	}
}

Adjacency Matrix

Another way we can represent a graph is by using an adjacency matrix. An adjacency matrix is a table which tells us if there is an edge between two vertices, and in the case of a weighted graph, the weight of the edge.

If there are N vertices in the graph, the adjacency matrix will be an N x N array of integers, where the rows represent the "from" end of the edge, and the columns represent the "to" end of the edge. The entry at (i, j) contains the weight of the edge from vertex #i to vertex #j, or infinity if no such edge exists. In the case of a bidirectional graph, if there is an edge from vertex #i to vertex #j there is also an edge from vertex #j to vertex #i, so the adjacency matrix will be symmetric.

Let's take another look at the graph we used for the adjacency list:

The adjacency matrix for this graph would be:

0 1 2 3 4 5 6 7
0 0 - 0 - - - - -
1 - 0 0 - - 0 - -
2 0 0 0 0 0 - - -
3 - - 0 0 - - - -
4 - - 0 - 0 - - -
5 - 0 - - - 0 0 0
6 - - - - - 0 0 -
7 - - - - - 0 - 0

Above, "-" means that there is no edge there (the value is infinity).

In this case, the graph was bidirectional, so if you look along the diagonal, you will see that the matrix is, in fact, symmetric. If this were a large graph with many vertices, we could save space by only storing the upper or lower triangle, and use those values as both index (i, j) and index (j, i).

Also, if you look at this matrix, you will see that we have 0's along the diagonal, which suggests that vertices are adjacent to themselves. This is implementation specific -- depending on what you are trying to represent, you may or may not want to let vertices be adjacent to themselves.

Now, let's take a look at code to implement an adjacency matrix. We will implement the same methods that the AdjList class has.

class AdjMatrix
{
	private int [][] matrix;
	private int num_vertices;

	private const int inf = Integer.MAX_VALUE;

	public AdjMatrix(int num_vertices)
	{
		this.num_vertices = num_vertices;
		matrix = new int[num_vertices][num_vertices];

		// let's assume that nodes are adjacent to themselves
		for (int vertex = 0; vertex < num_vertices; vertex++)
		{
			for (int other_vertex = 0; other_vertex < num_vertices; other_vertex++)
			{
				if (other_vertex == vertex)
				{
					matrix[vertex][other_vertex] = 0;
				}
				else
				{
					matrix[vertex][other_vertex] = inf;
				}
			}
		}
	}

	public void addEdge(int vertex_from, int vertex_to, int cost)
	{
		matrix[vertex_from][vertex_to] = cost;
	}

	public void addEdge(int vertex_from, int vertex_to)
	{
		addEdge(vertex_from, vertex_to, 0);
	}

	public void addBidirectedEdge(int vertex_one, int vertex_two, int cost)
	{
		addEdge(vertex_one, vertex_two, cost);
		addEdge(vertex_two, vertex_one, cost);
	}

	public void addBidirectedEdge(int vertex_one, int vertex_two)
	{
		addEdge(vertex_one, vertex_two, 0);
		addEdge(vertex_two, vertex_one, 0);
	}

	public removeEdge(int vertex_from, int vertex_to)
	{
		matrix[vertex_from][vertex_to] = inf;
	}

	public removeBidirectedEdge(int vertex_one, int vertex_two)
	{
		matrix[vertex_one][vertex_two] = inf;
		matrix[vertex_two][vertex_one] = inf;
	}

	public int getEdge(int vertex_from, int vertex_to)
	{
		return matrix[vertex_from][vertex_to];
	}
}
	

Lists Vs. Matrices

We have two ways to represent graphs, so which one should we use?

Well, if the graph is sparse (there aren't many edges), then the matrix will take up a lot of space indication all of the pairs of vertices which don't have an edge between them, but the adjacency list does not have that problem, because it only keeps track of what edges are actually in the graph. On the other hand, if there are a lot of edges in the graph, or if it is fully connected, then the list has a lot of overhead because of all of the references.

If we need to look specifically at a given edge, we can go right to that spot in the matrix, but in the list we might have to traverse a long linked list before we hit the end and find out that it is not in the graph.

On the other hand, if we need to look at all of a vertex's neighbors, if you use a matrix you will have to scan through all of the vertices which aren't neighbors as well, whereas in the list you can just scan the linked-list of neighbors.

If, in a directed graph, we ask the question, "Which verticies have edges leadingt to vertex X?", the answer is straight-forward to find in an adjacency matrix -- we just walk down column X and report all of the edges that are present. But, life isn't so easy with the adjacency list -- we actually have to perform a brute-force search.

So which representation you use depends on what you are trying to represent and what you plan on doing with the graph.

Topological Sort

A topological sort is an ordering of vertices in a directed, acyclic (no cycles) graph which travels in one direction only. In order to get a college degree, you must complete a topological sort (ordering) of the required courses, taking prerequisites before you take the coures which build on those prerequisites. Think of the core (or part of the core) curriculum for Information Science.

Which vertices have an indegree of 0? The freshman-level courses. Topological sort puts all vertices with an indegree of 0 into a queue.

36-201,	15-1xx,	21-121
  

While the queue is not empty, dequeue.

Dequeue 36-201 and remove it. What happens to the indegree of 36-202? It becomes 0, so enqueue it.

15-1xx,	21-121,	36-202
  

Dequeue 15-1xx and remove it. Now the indegree of 15-200 is 0, so enqueue it.

21-121,	36-202, 15-200
  

Dequeue 21-121 and remove it. Now the indegree of 21-122 is 0, so enqueue it.

36-202, 15-200, 21-121
  

Dequeue 36-202 and remove it. Now the indegree of 36-203 is 0, so enqueue it.

15-200, 21-121, 36-203
  

Dequeue 15-200 and remove it. This does not reduce the indegree of any vertex to 0.

21-121, 36-203
  

Dequeue 21-121 and remove it. Now the indegree of 66-270 is 0, so enqueue it.

36-203, 66-270
  

Dequeue 36-203. This does not reduce the indegree of any vertex to 0.

66-270
  

Dequeue 66-270 and remove it. Now the indegree of 66-271 is 0, so enqueue it.

66-271
  

Dequeue 66-271 and remove it. Now the indegree of 66-272 is 0, so enqueue it.

66-272
  

Dequeue 66-272 and remove it. Now the indegree of 66-273 is 0, so enqueue it.

66-273
  

Dequeue 66-273 and graduate!

How would you implement a topological sort of a graph? You'll need a queue. You'll also need to keep track of the indegree of each vertex in the graph so that you can enqueue vertices of indegree 0. You'll start your algorithm by enqueing every vertex in the graph with an indegree of 0, and your algorithm will terminate when the queue is empty.

Assuming that the indegree of each vertex is stored and that the graph has been read into an adjacency list, we can apply the following (pseudocode) algorithm to generate a topological ordering. Let's assume that the graph is directed and has no cycles.


  void topologicalSort()
  {
    Queue q; //keeps track of what vertex is next (e.g., what course to take next)
    Vertex v, w; //references to Vertex objects
    int counter=0; //keeps track of the topological ordering of the dequeued vertices (e.g., order of courses taken)

    q=new Queue();

    //enqueue all vertices with an indegree of 0 (e.g., courses with no unfulfilled prerequisites
    for each vertex v
      if (v.indegree == 0)
        q.enqueue(v);	
	
      //dequeue vertices one at a time, reducing the indegree of their children as you remove them
      while(!q.isEmpty())
      {
        v = q.dequeue();
        v.topNum == ++counter; //give the vertex you're dequeing its order in the topological sort
	    
        for each w adjacent to v
          if(--w.indegree == 0)
            q.enqueue(w);
      }
    }