...menustart
- Algorithm On Graphs
- Week1 Exploring Undirected Graphs
- Week2 Directed Graphs
- Week3 Shortest Path
- Week4 Weighted Edge Graph
- Week 5: Minimum Spanning Tree (MST)
- Week6 Advanced Shortest Path
...menuend
- Edge List
- egdes: (A, B), (A, C), (A,D), (C,D)
- A,B,C,D are vertices
- Adjacency Matrix
- Entries 1 if there is an edge, 0 if there is not.
- Adjacency List
- For each vertex, a list of adjacent vertices.
- A adjacent to B, C,D
- B adjacent to A
- C adjacent to A,D
- D adjacent to A, C
- can be implemented by a dictionary
- Different operations are faster in different representations
- for many problems , we want adjacency list because a lot of operations really want to be able to find neighbors.
Op | Is Edge? | List Edge | List Nbrs |
---|---|---|---|
Adj. Matrix | Θ(1) | Θ( |V|²) | Θ(|V|) |
Edge List | Θ(|E|) | Θ(|E|) | Θ(|E|) |
Adj. List | Θ(deg) | Θ(|E|) | Θ(deg) |
- Graph algorithm runtimes depend on |V| and |E|.
- i.e.
O(|V|+|E|)
- i.e.
- Visit Markers
- To keep track of vertices found: Give each vertex boolean visited(v).
- Unprocessed Vertices
- Keep a list of vertices with edges left to check.
- This will end up getting hidden in the program stack
- Depth First Ordering
- We will explore new edges in Depth First order.
- Explore(v) marks as visited exactly the vertices reachable from v.
- Good for adjacency list representation!
def Explore(v):
visited(v) ← true
for w in neighbors(v):
if not visited(w):
Explore(w)
- Normaly, we are used to save parent node for each vertex, so that we can easily reconstruct the path.
def Explore(v):
visited(v) ← true
for w in neighbors(v):
if not visited(w):
Explore(w)
edgeTo[w] = v
- DFS to find all vertices of G, not just those reachable from v.
def DFS(G):
for all v in V :
mark v unvisited
for v in V :
if not visited(v):
Explore(v)
- Explore(v) finds the connected component of v
- Just need to repeat to find other components.
- Modify DFS to do this.
- Modify goal to label connected components
- Solution: We also assign the vertices a number corresponding to the connected components
- Runtime: still O(|V|+|E|)
def Explore(v):
visited(v) ← true
# changes: cc is passed in outside
CCnum(v) ← cc
for w in neighbors(v):
if not visited(w):
Explore(w)
def DFS(G):
for all v in V :
mark v unvisited
# changes: generating connected components number
cc ← 1
for v in V :
if not visited(v):
Explore(v)
# changes: increment cc
cc ← cc + 1
- Need to Record Data
- Plain DFS just marks all vertices as visited.
- In general if we want ot make DFS useful, we need to keep track of other data to be useful, just like how we find the connected components.
- Adding functions to store additional information, for example, let's look at the
Explore(v)
def Explore(v):
visited(v) ← true
# changes
previsit(v)
for w in neighbors(v):
if not visited(w):
Explore(w)
# changes
postvisit(v)
So , what are those funcitons , previsit/postvisit going to be ? They could be many kind of things.
One that we might want to do is to keep track of what order are we visit vertices in. And so one way to do this is we have a clock.
- This clock keep track of order of visits.
- Clock ticks forward once it hit a pre-/post- visit.
- Records previsit and postvisit times for each v.
- Computing Pre- and Post- Numbers
- Initialize clock to 1.
def previsit(v):
pre(v) ← clock
clock ← clock + 1
def postvisit(v):
post(v) ← clock
clock ← clock + 1
- Previsit and Postvisit numbers tell us about the execution of DFS.
- Lemma
- For any vertices u, v the intervals pre(u), post(u)] and [pre(v), post(v)] are either nested or disjoint.
- nested: eg. (1,8) , (2,5)
- disjoint: eg. (1,8) , (9,12)
- that is , Interleaved is not possible
- eg. ( 1, 8 ) , ( 5, 9 )
- For any vertices u, v the intervals pre(u), post(u)] and [pre(v), post(v)] are either nested or disjoint.
- Directed graphs might be used to represent:
- Streets with one-way roads.
- Links between webpages.
- Followers on social network.
- Dependencies between tasks.
- Can still run DFS in directed graphs.
- Only follow directed edges
- explore(v) finds all vertices reachable from v.
- Can still compute pre- and post- orderings.
- A cycle in a graph G is a sequence of vertices v1, v2, . . . , vn so that
- (v1, v2),(v2, v3), . . . ,(vn−1, vn),(vn, v1) are all edges.
- Theorem
- If G contains a cycle, it cannot be linearly ordered.
- linearly ordered
- A directed graph G is a Directed Acyclic Graph (or DAG) if it has no cycles.
- Theorem
- Any DAG can be linearly ordered
- Beiing a DAG is necessary to linearly order.
- Is it sufficient ?
Goal. Given a set of tasks to be completed with precedence constraints, in which order should we schedule the tasks?
-
Learning Objectives
- Implement the topological sort algorithm
- Prove that a DAG can be linearly ordered
-
Last Vertex
-
Sources and Sinks
-
Idea
- Find sink
- Put at end of order
- Remove from graph
- Repeat
This is all well and good, but it depends on us being able to find a sink.
- Question: How do we know that there is a sink?
- Follow path as far as possible v1 → v2 → . . . → vn. Eventually either:
- Cannot extend (found sink).
- Repeat a vertex (have a cycle).
-
# LinearOrder(G ) while G non-empty: Follow a path until cannot extend Find sink v Put v at end of order Remove v from G
- Runtime
- O(|V|) paths
- Each takes O(|V|) time
- Runtime O(|V|²)
- Speed Up
LinearOrder
retrace same path every time- Instead only back up as far as necessary
- This is just DFS !
- And in particular whenever we finish the post visit block at a vertex,
TopologicalSort(G) DFS(G) sort vertices by reverse post-order
- Connectivity is complex in directed graph.
- Two vertices v, w in a directed graph are connected if you can reach v from w AND can reach w from v.
- Theorem
- Once we've split our graph into connected components, they still have edges connecting these components to each other
- So one useful thing to do is to draw what's known as metagraph, which sort of telles us how these strongly connected components connect to one another
- Theorem
- The metagraph of a (directed) graph G is always a DAG.
How to compute the strongly connected components of a graph. ?
- Problem
- Input: A directed graph G
- Output: The strongly connected components of G.
- Idea:
- If v is in a sink SCC, explore(v) finds vertices reachable from v. This is exactly the SCC of v.
- 因为SCC的性质,无论你从SCC中的任何一个点node开始,你都是 explore 整个SCC
- that also means, you will get different SCC, if you start from different node.
- If v is in a sink SCC, explore(v) finds vertices reachable from v. This is exactly the SCC of v.
- Need a way to find a sink SCC.
- why ?
- if you start from a source SCC, you many visit the whole graph, it won't help you.
- Theorem
- If C and C' are two strongly connected components with an edge from some vertex of C to some vertex of C' ,
- then largest postorder in C bigger than largest postorder in C' no matter you visit vertex first in C or C'.
- 但是,最小的 postorder 不一定在 C'里
- Conclusion
- The vertex with the largest postorder number is in a source component!
- Problem: We wanted a sink component
- Let Gᴿ be the graph obtained from G by reversing all of the edges.
- Gᴿ and G have same SCCs.
- Source components of Gᴿ are sink components of G.
- Find sink components of G by running DFS on Gᴿ .
- The vertex with largest postorder in Gᴿ is in a sink SCC of G.
-
def SCCs(G ): repeate: run DFS(Gᴿ) let v have largest post number run Explore(v) on G vertices found are first SCC , Remove from G
-
Improvement
- Don't need to rerun DFS on Gᴿ
- Largest remaining post number comes from sink component
- The point is, after you remove this sink component, and if you just then look for the vertex with the single largest remaining postorder, that's going to be sink component of the new graph.
-
def SCCs(G): run DFS(Gᴿ) for v ∈ V in reverse postorder: if not visited(v): Explore(v) mark visited vertices as new SCC
-
Runtime
- Essentially DFS on Gᴿ and then on G.
- Runtime O(|V| + |E|).
def BFS(G,S):
for all u∈V:
dist[u] ← ∞, prev[u] ← nil
dist[S] ← 0
Q ← {S} {queue containing just S}
while Q is not empty:
u ← Dequeue(Q)
for all (u,v)∈E:
if dist[v] = ∞:
Enqueue(Q , v )
dist[v] ← dist[u] + 1
prev[v] ← u
Here, we use dist[]
to save the accumulative distance, it also servers as visited
.
Normally the application of DFS, BFS visits each node only once.
By BFS, you have found the shortest path , but how you reconstuct the shortest path ?
def ReconstructPath(S, u, prev):
result ← empty
while u ≠ S:
result.append(u)
u ← prev[u]
return Reverse(result)
How to compute shortest path of a weighted edges graph ?
Note: normally there are 2 versions of Dijkstra. The key difference is whether a vertex is allowed be enqueued more than once. If re-enqueued is allowed, that version algorithm can work with negative weight.
- We've know how to find shortest path in the graph with non-negative edge weight.
- Now we will learn to find shortest path even in the graphs where some of the edge weigths can be negative.
We will explore that using a problem about currency exchange, which doesn't seem to have anything to do with shortest paths.
But we will soon find out it does.
-
initial: $1000
-
convert it into Russian Rubles.
-
Potentially doing many conversions on the way, such that you get as many Russian rubles as you can in the end.
-
- These triangular arbitrages exist very rarely and only due to some market inefficients.
- But in theory, this is possible.
-
The graph edge is conversion rate.
- use 2 standard approaches
- Replace product with sum by taking log() of weights
- Negate weights to solve minimization instead maximization
- Taking the logarithm
- xy = 2log₂(x) 2log₂(y) = 2log₂(x)+log₂(y)
- xy → max ⇔ log₂(x)+log₂(y) → max
- So, if all weights are positive , then we reduce our problem of maximizing product of some numbers to maximizing sum of some numbers.
- because we can not take log() of negative values, and we also can not take log() of 0. ( value of exponential function is positive )
- But we actually want minimization ... That is easy
- is is the same as minimize the sum of -value
- Finally: replace edge weights reᵢ by -log( reᵢ ) , and find the shortest path between USD and RUR in the graph.
- Solved ? Can we now apply Dijkstra algorithm to find the solution ?
- NOT exactly work.
- Dijkstra's algorithm rely on the fact that a shortest path from s to t goes only through vertices that are close to s.
- And this is rely on the fact that edge weights are positive
- This is no longer the case for graphs with negative edges
- Bellman-Ford Algorithm works even for negative edges weights.
- This algorithm assumes that there are no negative weight cycles in G.
- Otherwise it works still, but it won't return correct distances for some of the nodes.
def BellmanFord(G , S ):
{no negative weight cycles in G}
for all u∈V:
dist[u] ← ∞
prev[u] ← nil
dist[S] ← 0
repeat |V|−1 times:
for all (u,v)∈E:
Relax(u, v )
- The running time of Bellman–Ford algorithm is O(|V||E|).
- During the iterations, if no edge was actually relaxed , we can just stop there and the distance will already be correct.
- Input: A connected, undirected graph G = (V,E) with positive edge weights.
- Output: A subset of edges E′ ⊆ E of minimum total weight such that the graph (V , E′) is connected.
- A tree is an undirected graph that is connected and acyclic
- A tree on n vertices has n-1 edges
- Any connected undirected graph G(V,E) with |E| = |V|-1 is a tree
- An undirected graph is a tree iff there is a unique path between any pair of its vertices.
Algorithm | description |
---|---|
Kruskal | repeatedly add the next lightest edge if this doesn't produce a cycle |
Prim | repeatedly attach a new vertex to the current tree by a lightest edge |
-
Example
- Kruskal's algorithm
- Initially, our soulution is empty, so we just add the first lightest edge ( with weight 1)
- And there is also another edge with has weight 1, so we also add it to our solution.
- The next 1 is weight 2, we add it.
- The next lightest available edge has weight 3,
- The next lightest available edge has weight 4. Wee add it.
- Then, again, we try to add the edge with weight 5, however it produces a cycle, so we skip this edge, and instead we add the edge of weight 6.
- This gives us a solution.
- Prim's algorithm
- It works in a different way, it repeatedly grows just 1 tree. It will select a root for this tree. At each iteration we are going to attach a new node to this tree.
-
They start with an empty solution , with an empty set of edges. And at each iteration they expand the current set of edges by one edge.
- But they use different stratedies for selecting the next edge.
- The Kruskal's Algorithm selects the next lighted edge that doesn't produce a cycle.
- While the Prim's Algorithm selects the next lightest edge that attaches a new vertex to the current tree.
-
Let X ⊆ E be a part of a MST of G(V,E), S ⊆ V be such that no edge of X crosses between S and V − S, and e ∈ E be a lightest edge across this partition. Then X + {e} is a part of some MST.
- X is edges part of MST
- we partition all vertices into 2 parts -- S , and V-S
- this partition is required to satisfies the following property:
- e is the lightest edge connect the verticese across S and V-S
- Then if we add the edge e to our current X , then what we get will also be a subset of some minimum spanning tree.
- In other words, adding e to X in this case is a safe move.
-
Example
- We are given some subset of edges of a MST, shown here in blue. It is the set X.
- So X is a subset of some MST.
- Then consider some partition vertices into 2 parts
- Next we consider the lightest edge that joins 2 vertices from different parts, assume that this edge is e, shown here in red
- Then what lemma states is that if we add this edge to our set X, then the resulting set of edges will also be a part of some MST.
- repeatedly add to X the next lightest edge e that doesn’t produce a cycle
- At any point of time, the set X is a forest, that is, a collection of trees
- a single isolated vertex is also a tree
- The next edge e connects two different trees , say, T₁ and T₂
- if not , it produe a cycle
- The edge e is the lightest between T₁ and V − T₁, hence adding e is safe
- the cut property
- use disjoint sets data structure
- initially, each vertex lies in a separate set
- each set is the set of vertices of a connected component
- that is, initially, each vertex forms a separate connected component
- to check whether the current edge {u,v} produces a cycle, we check whether u and v belong to the same set -- in the same connected component
- and also, if they are lie in different connected components, we need to merge the corresponding 2 trees, union 2 sets
def Kruskal(G ):
for all u∈V:
MakeSet(v )
X ← empty set
sort the edges E by weight
for all {u,v}∈E in non-decreasing weight order:
if Find(u) ≠ Find(v):
add {u,v} to X
Union(u, v )
return X
O(|E|·log|V|)
- X is always a subtree, grows by one edge at each iteration
- we add a lightest edge between a vertex of the tree and a vertex not in the tree
- very similar to Dijkstra’s algorithm
- Prim is greedy, it does not care about the accumulative cost, it just use the edge weight for priority
def Prim(G ):
for all u∈V:
cost[u] ← ∞, parent[u] ← nil
pick any initial vertex u0
cost[u0] ← 0
PrioQ ← MakeQueue(V ) {priority is cost}
while PrioQ is not empty:
v ← ExtractMin(PrioQ)
for all {v,z} ∈ E:
if z ∈ PrioQ and cost[z] > w(v, z):
cost[z] ← w(v, z), parent[z] ← v
ChangePriority(PrioQ , z , cost [z ])
- What problem is this algorithm solved?
- The shortest path between source vertex s and a target vertex t.
- Yes, we have a goal.
- Why not just Dijkstra ?
- It is pretty fast, right ?
- Can we even do better in the worst case ?
- Yes, Mikkel Thorup came up with an algorithm in 19999 that solves this problem for undirected graphs in linear time.
- Such algorithm is still not known for directed graph. But for undirected graphs, we can definitely do better, and for directed graphs, we can do better also.
- For a graph of USA with 20M vertices and 50M edges it will work for serveral seconds for average.
- Millions of users of Google Maps want the result in a blink of an eye, all at the same time.
- Need something significantly faster
-
Instead of going from S and growing the "search circle" until it touches point, we want to go simultaneously forward from S , and backward from T until we meet.
-
And as soon as we meet, we can find the shortest path between S and T by combining the half of the path S->Mid-point and Mid-point->T.
-
We will alternate the turns of those 2 Dijkstra.
-
And now, we can reconstruct the shortest path from S to T, and we can fill in the correct distances from S to all the nodes on that path.
-
Roughly 2x speedup
- Good, but not great
- Applying the idea of bidirectional search to social network data, we will see that it will work exceptionally well, going up to thousands of times faster.
- But first , let's learn an interesting fact about our world.
- In 1929, Hungarian mathematician Frigyes Karinthy made a "Small world" conjecture :
- Can pass a message from any person to any person in at most 6 handshakes.
- This is close to truth according to experiments and it called a "six handshakes" or "six degrees of separation" idea.
- Meet-in-the-middle
- More general idea, not just for graphs
- Instead of searching for all possible objects, search for 1st halves and for 2nd halves separately
- Then find "compatible" havles
- √N instead of N
- Build Gᴿ
- Start Dijkstra from s in G , and from t in Gᴿ
- Alternate between Dijkstra steps in G and in Gᴿ
- Stop when some vertex v is processed both in G and in Gᴿ
- process means that it is extracted from the PQ
- Compute the shortest path between s and t
-
Let v be the 1st vertex which is processed both in G and Gᴿ. Can we guarantee then that there exists a shortest path from s to t that goes through v?
-
No.
-
Lemma
- Let dist[u] be the distance estimate in the forward Dijkstra from s in G
- and distᴿ[u] -- the same in the backward Dijkstra from t in Gᴿ.
- After some node v is processed both in G and Gᴿ,
- there exists some shortest path from s to t passes through some node u which is processed either in G, in Gᴿ, or both, and d(s,t) = dist[u] + distᴿ[u].
- Let dist[u] be the distance estimate in the forward Dijkstra from s in G
- Implementation
- Do alternating turns of forward search and backward search , until we meet at some point v
- and we remember which vertices are processed in forward search and which are in backward search
- Then we take all those vertices which are processed at least in one of them , and for each of those vertices, we minimize the sum of distance estimate of d[u]+dᴿ[u],
- and for the node for which this sum is minimal , we know that there is a shortest path going through this vertex.
- And then to reconstruct the path itself
- We can reconstruct the shorteset path from s to u in forward search, and separately we reconstruct the path from u to t in backward search.
- and then just join those 2 parts into a single shortest path from s to t.
- Do alternating turns of forward search and backward search , until we meet at some point v
def BidirectionalDijkstra(G , s , t ):
Gᴿ ← ReverseGraph(G) # create reverse graph
Fill dist,distᴿ with +∞ for each node # init dist
dist[s] ← 0, distᴿ [t] ← 0
Fill prev,prevᴿ with None for each node # prev vertex
proc ← empty, procᴿ ← empty # processing vertices
while True
v ← ExtractMin(dist)
Process(v , G , dist, prev, proc)
if v in procᴿ:
return ShortestPath(s, dist, prev, proc, t, . . . )
vᴿ ← ExtractMin(distᴿ)
repeat symmetrically for vᴿ as for v
def Relax(u, v , dist, prev):
if dist[v]>dist[u]+w(u,v):
dist[v] ← dist[u] + w(u, v)
prev[v] ← u
def Process(u, G , dist, prev, proc):
for (u,v)∈E(G):
Relax(u, v , dist, prev)
proc.Append(u)
def ShortestPath(s, dist, prev, proc, t, distᴿ , prevᴿ , procᴿ ):
distance ← +∞, ubest ← None
for u in proc+procᴿ: # union, at least in one set
if dist[u]+distᴿ[u]<distance:
ubest ← u
distance ← dist[u] + distᴿ [u]
path ← empty
last ← ubest
while last≠ s:
path.Append(last)
last ← prev[last]
path ← Reverse(path)
last ← ubest
while last≠ t:
last ← prevᴿ[last]
path.Append(last)
return (distance,path)
- Worst-case running time of Bidirectional Dijkstra is the same as for Dijkstra
- Speedup in practic depends on the graph , e.g. social network graph
- Memory consumption is 2x to store G and Gᴿ
- Bidirectional Dijkstra can be 1000s of times faster than Dijkstra for social networks
- But just 2x speedup for road network
- This lecture -- great speedup for road networks
- Long-distance trips go through highways
- Less important roads merge into more important roads
- Hierarchies of roads
- So the idea is to use the structure of the shortest paths in the real road networks,
- this will allow us to acutally avoid scanning many small roads, many not important vertices in the graph
- If you go from San Francisco to New York, then most probably you don't need to go through small streets somewhere in Las Vegas or Chicago. Most of the way, you'll be going through a big highway.
- There are algorithms based on this idea
- “Highway Hierarchies” and “Transit Node Routing” by Sanders and Schultes
- Millions of times faster than Dijkstra
- Pretty complex
- This lecture — “Contraction Hierarchies”, thousands of times faster than Dijkstra
- Nodes can be ordered by some “importance”
- The idea of the contraction hierarchies algorithm is to order the nodes by importance.
- Importance first increases, then decreases back along any shortest path
- E.g., points where a highway merges into another highway
- Can use bidirectional search
- Preprocess the graph
- It can be a long process, It can take a few hours or even days.
- Find distance and shortest path in the preprocessed graph
- But then when you're ready , and you've saved the results of your preprocessing, you can answer the queries for distance and shortest paths much faster.
- Reconstruct the shortest path in the initial graph
- Eliminate nodes one by one in some order
- when we eliminate the node, some of the shortest paths that existed in the initial graph can be gone because they were passing through this node.
- Add shortcuts to preserve distances
- and in this case we need to add some shortcuts so that we preserve the distance.
- so that the distances between any 2 nodes that are still in the graph , are the same as the distances in the initial graph.
- so we'll add some shortcuts , some new edges to the graph
- Output: augmented graph + node order
- that augmented graph is the graph with augmented set of edges
- it has the same set of vertices as the initial graph.
- it also has all the edges of the initial graph.
- but apart from that it has all the added shortcuts as edges.
- and also we'll output the order of the nodes that we used in this preprocessing.
- that augmented graph is the graph with augmented set of edges
Let's look at this very simple line graph.
All nodes in one chain. The nodes are numbered ,and already in the order , in which we going to contract the nodes in this example.
Let's look what happens when we contract the nodes. First, we contract node 1, it goes down, means that we contracted it.
And you see that there was a path 6-1-4 , when we contracted it, it's no longer in the graph . And then we need to reconstruct that path. So we add a new edge 6-4 with length 5, which is colored in blue in the picture.
Now what happens when we contract node 2 ? Nothing really happens.
Now let's contract node 3.
When we contract the node 4, it is the most interesting node because it already has 2 blue nodes. But nevertheless, when we contract it we remove the path between 6-4-5, and add a new edge 6-5.
In the end we contract node 5. Nothing changes.
We don't need to contract node 6 because it's the last node in the graph.
You see that the nodes in the new picture are different heights. And this just symbolizes that the higher is the node, the later it was contracted.
And the higher the node , the more important it is. So we first contract or remove the least important nodes. And the nodes which are left in the end are the most important nodes.
Now let's see what happens in general when we contract node v.