Graph theory in Computer competition
Intro
Graph theory is prabobly the most popular question in computer competition in USACO and NOIP(China). It is actually a branch of math theory. In the questions they would give you many points in a graph and ask you to find out the shortest way or the possile way to go from one point to another one. Sometimes it will provide you with values of the edge and ask you to calculate the smallest coast. Sounds pretty simple Hahhhhh. Let’s have a look!
Basic Information
PS: You can jump to next block and keep reading, and find the concept you don’t understand above.
Graph is a 2-tuple which describes the relationship between the points. Usually explained as G=(V,E), and V, E are mutiple sets.
Graph has two different types:
Undirected graph: Every elements in E are disordered 2-tuple(u,v), we name the element Undirected edge. And u,v$\in$V. Let e=(u,v), then u and v are e’s endpoint
Directed graph: Every elements in E are ordered 2-tuple, write as u$\longrightarrow$v, we name the element Directed edge. Let e=u$\longrightarrow$v, then u is e’s Tail, and v is e’s head.
And for every elements in V, we call it Vertex or Node.
…….
Actually there is a lot more concept we need to know for the Graph Theory, but we are just programmers(I don’t want to write down) Here is the link for More basic info.
Restore the graph!
After we study the basic information, know we need to know how to restore those information about a graph in our computer.
There are three ways we use to restore graphs.
But to make it easy(I have no idea how to use the other two), I will only show the one that is most easy(Even can be mastered by the damn author)
Actually, Forward star can widely use on different kind of graphs, this is the main reason I introduce it.
Forward Star
We first use Struct to store the information about the graph.
We need to restore the basic info in the graph. Based on edge. So here are the info we need to restore in the struct: The value of the edge, the point the edge go to, the next edge.
So here is the code:
1 | struct Edge{ |
Then we add the data into the struct
1 | void addedge(int u, int v, int val){ |
And here is the code in main function:
1 | int main(){ |
This way we make the restorage logical and easy. Then we can actually do something with the theory.
Graph traversal
Find the relationship between two points in a graph can be seen as travel in the graph. And we need to know the basic how to travel through the whole graph.
1 | void travel(int s){//Starting point |
But this kind of method is slow and do not fit most of the situation. We need some spcial methods.
Then here are two big basic alograithm for Graph theory.
DFS
DFS, full name Depth First Search.
To make it easy, the core of it is to follow one road till it is going to an end. We travel through every edges as long as the edge can be traveled through.
1 | int vis[mx]={0};//To make sure you won't travel through the same edge over and over again |
Simple as this.
With the core of the dfs, we can solve some problems easily.
[USACO06DEC] Cow Picnic
Description
The cows are having a picnic! Each of Farmer John’s K (1 ≤ K ≤ 100) cows is grazing in one of N (1 ≤ N ≤ 1,000) pastures, conveniently numbered 1…N. The pastures are connected by M (1 ≤ M ≤ 10,000) one-way paths (no path connects a pasture to itself).
The cows want to gather in the same pasture for their picnic, but (because of the one-way paths) some cows may only be able to get to some pastures. Help the cows out by figuring out how many pastures are reachable by all cows, and hence are possible picnic locations.
Input
Line 1: Three space-separated integers, respectively: K, N, and M
Lines 2..K+1: Line i+1 contains a single integer (1..N) which is the number of the pasture in which cow i is grazing.
Lines K+2..M+K+1: Each line contains two space-separated integers, respectively A and B (both 1..N and A != B), representing a one-way path from pasture A to pasture B.
output
Line 1: The single integer that is the number of pastures that are reachable by all cows via the one-way paths.
Sample
Input
1 | 2 4 4 |
Output
1 | 2 |
Solution:
Dfs search every cows, use field[i] express the numer i field have been tracel through how many times, if field[i]=k then meet the standard(Means every cow has traveled through here).
1 |
|
BFS
BFS, full name Breadth First Search
BFS is a traversing algorithm where you should start traversing from a selected node (source or starting node) and traverse the graph layerwise thus exploring the neighbour nodes (nodes which are directly connected to source node). You must then move towards the next-level neighbour nodes.
Code:
1 |
|
Example
[USACO14DEC] Piggy Back
Description
Bessie and her sister Elsie graze in different fields during the day, and in the evening they both want to walk back to the barn to rest. Being clever bovines, they come up with a plan to minimize the total amount of energy they both spend while walking.
Bessie spends B units of energy when walking from a field to an adjacent field, and Elsie spends E units of energy when she walks to an adjacent field. However, if Bessie and Elsie are together in the same field, Bessie can carry Elsie on her shoulders and both can move to an adjacent field while spending only P units of energy (where P might be considerably less than B+E, the amount Bessie and Elsie would have spent individually walking to the adjacent field). If P is very small, the most energy-efficient solution may involve Bessie and Elsie traveling to a common meeting field, then traveling together piggyback for the rest of the journey to the barn. Of course, if P is large, it may still make the most sense for Bessie and Elsie to travel
separately. On a side note, Bessie and Elsie are both unhappy with the term “piggyback”, as they don’t see why the pigs on the farm should deserve all the credit for this remarkable form of
transportation.
Given B, E, and P, as well as the layout of the farm, please compute the minimum amount of energy required for Bessie and Elsie to reach the barn.
Input
INPUT: (file piggyback.in)
The first line of input contains the positive integers B, E, P, N, and M. All of these are at most 40,000. B, E, and P are described above. N is the number of fields in the farm (numbered 1..N, where N >= 3), and M is the number of connections between fields. Bessie and Elsie start in fields 1 and 2, respectively. The barn resides in field N.
The next M lines in the input each describe a connection between a pair of different fields, specified by the integer indices of the two fields. Connections are bi-directional. It is always possible to travel from field 1 to field N, and field 2 to field N, along a series of such connections.
output
OUTPUT: (file piggyback.out)
A single integer specifying the minimum amount of energy Bessie and
Elsie collectively need to spend to reach the barn. In the example
shown here, Bessie travels from 1 to 4 and Elsie travels from 2 to 3
to 4. Then, they travel together from 4 to 7 to 8.
Sample
Input:
1 | 4 4 5 8 8 |
Output
1 | 22 |
Solution:
Because the shortest distance from one point to the destination is unchangable, so we don’t have to consider two of them meet together and depart. If B+E*<=P , then we should keep them being together;
If B+E >*P Then they’d better be apart.
So we only need to run bfs from 1,2,n. And get the shortest length. After we enumerate the points they will be together, we are done.
1 |
|
More advanced Algotithm
Dijkstra
It uses the idea of BFS, aims to solve the Shortest path problem. It works both well for Directed graph and undirected graph.
The point of Djikstra is be greedy. We declare an array to restore the distance from the starting point to another point, initialize them as infinite number if there is no direct edge from starting point to them, and if they do have, put them as the smae value as the value of the edge .So that we can replace them with smaller number which we gained from the value of the route. As the graph.
Dijkstra has Time complexity 𝑂(n^2), but with heap, it can be 𝑂((𝑛+𝑚)log2𝑛)
1 | struct node{//Declare another struct to restore the information for the point |
[USACO19DEC]Milk Pumping G
Description
Farmer John has recently purchased a new farm to expand his milk production empire. The new farm is connected to a nearby town by a network of pipes, and FJ wants to figure out the best set of these pipes to purchase for his use in pumping milk from the farm to the town.
The network of pipes is described by NN junction points (endpoints of pipes), conveniently numbered 1…N1…N (2≤N≤10002≤N≤1000). Junction point 1 represents FJ’s farm and junction point NN is the town. There are MM bi-directional pipes (1≤M≤10001≤M≤1000), each joining a pair of junction points. The iith pipe costs cici dollars for FJ to purchase for his use, and can support a flow rate of fifi liters of milk per second.
FJ wants to purchase a single path worth of pipes, where the endpoints of the path are junctions 1 and NN. The cost of the path is the sum of the costs of the pipes along the path. The flow rate along the path is the minimum of the flow rates of the pipes along the path (since this serves as a bottleneck for the flow traveling down the path). FJ wants to maximize the flow rate of the path divided by the cost of the path. It is guaranteed that a path from 11 to NN exists.
Input
The first line of input contains NN and M.M. Each of the following MM lines describes a pipe in terms of four integers: aa and bb (the two different junctions connected by the pipe), cc (its cost), and ff (its flow rate). Cost and flow rate are both positive integers in the range 1…10001…1000.
output
Please print 106106 times the optimal solution value, truncated to an integer (that is, rounded down to the next-lowest integer if this number is not itself an integer).
Sample
Input:
1 | 3 2 |
Output
1 | 428571 |
Solution:
Just enumerate the biggest flow from 1 to 1000, and calculate for the smallest cost.
1 |
|
Floyd
It solve the Shortest path problem between two points, unlike Dijkstra, Floyd allow us to deal with the edge which has negative value. But Floyd take more time complexity(O(n^3)). So we have to be careful about it.
Like Dijkstra, Floyd’s core is also enumerate. It enumerate every point by 3 for loops. Because we suppose there is a shorter way than directly get to the point. We use g[i] [j] to express the distance from i to j, so there might be a point k in between might make this case:
1 | g[i][j]>g[i][k]+g[k][j]; |
I think it is easier to understand by code:(Probably the shortest one you ever meet)
1 | void floyd(){ |
[USACO09DEC]Cow Toll Paths
Description
Like everyone else, FJ is always thinking up ways to increase his revenue. To this end, he has set up a series of tolls that the cows will pay when they traverse the cowpaths throughout the farm.
The cows move from any of the N (1 <= N <= 250) pastures conveniently numbered 1..N to any other pasture over a set of M (1 <= M <= 10,000) bidirectional cowpaths that connect pairs of different pastures A_j and B_j (1 <= A_j <= N; 1 <= B_j <= N). FJ has assigned a toll L_j (1 <= L_j <= 100,000) to the path connecting pastures A_j and B_j.
While there may be multiple cowpaths connecting the same pair of pastures, a cowpath will never connect a pasture to itself. Best of all, a cow can always move from any one pasture to any other pasture by following some sequence of cowpaths.
In an act that can only be described as greedy, FJ has also assigned a toll C_i (1 <= C_i <= 100,000) to every pasture. The cost of moving from one pasture to some different pasture is the sum of the tolls for each of the cowpaths that were traversed plus a single additional toll that is the maximum of all the pasture tolls encountered along the way, including the initial and destination pastures.
The patient cows wish to investigate their options. They want you to write a program that accepts K (1 <= K <= 10,000) queries and outputs the minimum cost of trip specified by each query. Query i is a pair of numbers s_i and t_i (1 <= s_i <= N; 1 <= t_i <= N; s_i != t_i) specifying a starting and ending pasture.
Consider this example diagram with five pastures:
The ‘edge toll’ for the path from pasture 1 to pasture 2 is 3. Pasture 2’s ‘node toll’ is 5.
To travel from pasture 1 to pasture 4, traverse pastures 1 to 3 to 5 to 4. This incurs an edge toll of 2+1+1=4 and a node toll of 4 (since pasture 5’s toll is greatest), for a total cost of 4+4=8.
The best way to travel from pasture 2 to pasture 3 is to traverse pastures 2 to 5 to 3. This incurs an edge toll of 3+1=4 and a node toll of 5, for a total cost of 4+5=9.
Input
* Line 1: Three space separated integers: N, M, and K
* Lines 2..N+1: Line i+1 contains a single integer: C_i
* Lines N+2..N+M+1: Line j+N+1 contains three space separated
integers: A_j, B_j, and L_j
* Lines N+M+2..N+M+K+1: Line i+N+M+1 specifies query i using two space-separated integers: s_i and t_i
output
* Lines 1..K: Line i contains a single integer which is the lowest cost of any route from s_i to t_i
Sample
Input:
1 | 5 7 2 |
Output
1 | 8 |
Solution:
N so small and asking so many. Definally Floyd. We use two arries dis[][] and dist[][] to memorize the value of the edges alone and the value of the edges add with the value of points. We enumerate the value of points from small to big for point k, then the last k we enumerate should have the biggest point value from a to b. And we compare it with i and j’s value we then get the biggest point value. Then we are done.
1 |
|
SPFA
SPFA, full name Shortest Path Faster Algorithm.
Sounds pretty cool Hahhh. Actually, I think it is the most useful algorithm for graph theory, it can deal with almost all kinds of graph. Now, let’s see how it works!
It works by repeatedly selecting a vertex and using it to relax, if possible, all of its neighbors. If a node was successfully relaxed, then it might in turn be necessary to use it to relax other vertices, and hence it is marked for consideration also. Once there are no vertices left to be considered, the algorithm terminates. Note that a vertex might be considered several times during the course of the algorithm. The usual implementation strategy is to use a queue to hold the vertices that might be considered next.
Code:
1 | void spfa(int s){ |
We see that spfa is pretty similar to Dijkstra. let me use the core code to express the difference.
Difference between Dij+heap and SPFA!!!
Dij:
1 | while(!q.empty()){ |
SPFA:
1 | while(!q.empty()){ |
So the difference is clear enough:
Dji+heap: Small root pile, every time get the shortest distance, for this point, the shortest distance won’t change!
SPFA: Use queue. Get the front out of queue, might be renew in the future, it is won’t be always the same.
[USACO14OPEN]Dueling GPS’s
Description
Farmer John has recently purchased a new car online, but in his haste he accidentally clicked the “Submit” button twice when selecting extra features for the car, and as a result the car ended up equipped with two GPS navigation systems! Even worse, the two systems often make conflicting decisions about the route that FJ should take.
The map of the region in which FJ lives consists of N intersections (2 <= N <= 10,000) and M directional roads (1 <= M <= 50,000). Road i connects intersections A_i (1 <= A_i <= N) and B_i (1 <= B_i <= N). Multiple roads could connect the same pair of intersections, and a bi-directional road (one permitting two-way travel) is represented by two separate directional roads in opposite orientations. FJ’s house is located at intersection 1, and his farm is located at intersection N. It is possible to reach the farm from his house by traveling along a series of directional roads.
Both GPS units are using the same underlying map as described above; however, they have different notions for the travel time along each road. Road i takes P_i units of time to traverse according to the first GPS unit, and Q_i units of time to traverse according to the second unit (each travel time is an integer in the range 1..100,000).
FJ wants to travel from his house to the farm. However, each GPS unit complains loudly any time FJ follows a road (say, from intersection X to intersection Y) that the GPS unit believes not to be part of a shortest route from X to the farm (it is even possible that both GPS units can complain, if FJ takes a road that neither unit likes).
Please help FJ determine the minimum possible number of total complaints he can receive if he chooses his route appropriately. If both GPS units complain when FJ follows a road, this counts as +2 towards the total.
Input
* Line 1: The integers N and M.
Line i describes road i with four integers: A_i B_i P_i Q_i.
output
* Line 1: The minimum total number of complaints FJ can receive if he routes himself from his house to the farm optimally.
Sample
Input:
1 | 5 7 |
Output
1 | 1 |
Solution:
At first we need to know the shortest path for both GPS. And we need to know every point’s distance from the end point n, there is no way we need to use other alogrithm, we can just run sofa from n. How can we calculate how many time the GPS complained? We can simplely enumerate every edge, to see if its value is equal to the head and tail’s shortest length, and use dist[] to memorize how many time will the GPS complain if we across this route. Then we creat a new map based on this dist[], and spfa find the smallest path.
1 |
|
Conclusion
In this blog we studied DFS, BFS, Dijkstra, Floyd, and SPFA. They have their own advantages so we need to use them properly in competition. Here are a chart about each. Hopefully this woould be helpful to you.
Usage
Shortest-PathsProblem | Sparse Graph | Dense Graph | With negative value |
---|---|---|---|
Single-Source | Dijkstra+heap | SPFA/Dijkstra+heap | SPFA |
APSP(Undirected graph) | SPFA/Floyd | SPFA | SPFA |
APSP(Directed graph) | Floyd | SPFA/Dijkstra+heap | SPFA |
APSP((All Pairs Shortest Path))
Complexity
Solving ways | Time Complexity | Space Complexity |
---|---|---|
Dijkstra+heap | O(E*lgV) | O(n^2) |
SPFA | O(kE) (Not stable) | O(n) |
Floyd | O(n^3) | O(n) |
Reference Links:
1.https://www.luogu.com.cn/problem/P3110
2.https://www.luogu.com.cn/problem/P2853
3.https://en.wikipedia.org/wiki/Depth-first_search
4.https://en.wikipedia.org/wiki/Breadth-first_search
5.https://en.wikipedia.org/wiki/Shortest_Path_Faster_Algorithm
6.https://en.wikipedia.org/wiki/Floyd%E2%80%93Warshall_algorithm
7.https://en.wikipedia.org/wiki/Dijkstra%27s_algorithm
8.https://blog.csdn.net/qq_35644234/article/details/60870719
9.http://www.usaco.org/index.php?page=viewproblem2&cpid=969&lang=en
10.https://www.luogu.com.cn/problem/P2966
11.http://wcipeg.com/wiki/Shortest_Path_Faster_Algorithm