There are n courses — course 0 to n-1. Some of them have prerequisites. For example, courses A and B must be completed before course C can be taken (in other words, course C depends on A and B).
Find and return an order in which all the given courses can be taken, while satisfying all the prerequisites. If there exists more than one such order, any one of them would be a correct answer. If no such order exists, return a special value: [-1].
Input: n=4, prerequisites=[[1, 0], [2, 0], [3, 1], [3, 2]]
Output: [0, 1, 2, 3]
According to the input:
● Course 0 must be done before both 1 and 2
● Courses 1 and 2 must be done before course 3
There are two orders in which one can take all four courses: [0, 1, 2, 3] and [0, 2, 1, 3]. Both are correct answers.
Function accepts two arguments: The number of courses n and the list of prerequisites.
Prerequisites are given in the form of a two-dimensional array (or a list of lists) of integers. Each inner array has exactly two elements — it is essentially a list of pairs. Each pair [X, Y] represents one prerequisite: Course Y must be completed before X X depends on Y).
Function must return an array (list) of integers.
If all given courses can be taken while satisfying all given prerequisites, the returned array must contain a possible ordering (if more than one such ordering exists, any one must be returned). Otherwise, the function must return an array (list) with one element -1 in it.
This is a topological sorting problem. We’ll cover two efficient sample solutions: one uses DFS, while the second keeps track of the in-degree of the graph nodes. They have the same time and space complexity in the Big-O notation in terms of the number of courses and the number of prerequisites.
Both algorithms need a directed graph to work with. So, let us build one.
For example, if n=4 and prerequisites=[[1, 0], [2, 0], [3, 1], [3, 2]], the graph would look like this:
The correct answer to the problem will be a topological order of the nodes. The two sample solutions discussed below are essentially two different implementations of the topological sort algorithm.
Topological ordering doesn’t exist if the graph has a cycle. Both sample solutions detect cycles in their respective ways and return the special value in that case.
Topological sort implementation via DFS goes like this:
That’s what dfs_solution.cpp does.
It uses a standard DFS approach for detecting cycles: as we run DFS, if we encounter an edge from the “current” node to one that is visited but isn’t finished with, that means there is a cycle.
O(n + e), where n is the number of courses/nodes, and e is the number of prerequisites/edges.
Top-level DFS takes O(n + e) time. On top of that, we:
O(n + e)
The graph we build takes O(n + e) space, and the DFS stack also takes up to O(n + e) space.
O(n + e)
Input takes O(e) space, output takes O(n), and the auxiliary space used is O(n + e).
In-degree of a node in a directed graph is the number of edges that end in that node. This approach to topological sorting is based on the observation that any node with zero in-degree can be readily added to the topological ordering. The same applies to the courses and dependencies: we can always start taking courses from one with no dependencies.
So, we first add any node with zero in-degree to the “answer” list. Then we “erase” that node from the graph.
Note: While conceptually we remove a node and edges that start from it from the graph, in the actual sample solution, modification of the “edges” list happens to be unnecessary because of how “in_degree” and “nodes_with_zero_in_degree” lists are used.
As a result, any courses dependent on that “erased” node will not have that dependency. If that was their last dependency, they now have none — so zero in-degree — and can be added to the “answer” in the next round.
We repeat this (picking any node with zero in-degree, adding it to the “answer” list, removing it from the graph) until we run out of nodes with zero in-degree, and we are done. “in_degree” list is used to locate nodes with zero in-degree efficiently (saving us a loop over the remaining nodes to find one with zero in-degree any time we need one).
Cycle detection: If we run out of nodes with zero in-degree while some nodes still remain in the graph, it means that the remaining nodes all have incoming edges. If we visually imagine such a portion of a graph, we can see that it necessarily has a cycle.
O(n + e), where n is the number of courses/nodes, and e is the number of prerequisites/edges.
Building the graph from input data takes O(n + e) time.
The main part of the solution consists of a FOR loop inside of a WHILE loop. Body of the WHILE loop is executed n times at worst, while the body of the FOR loop is executed e times at worst (even though the FOR loop itself is “attempted” up to n times). So, O(n + e) is the total.
The intermediate data structures for representing the graph and for tracking the in-degree of nodes take O(n + e) space.
O(n + e).
There are n courses — course 0 to n-1. Some of them have prerequisites. For example, courses A and B must be completed before course C can be taken (in other words, course C depends on A and B).
Find and return an order in which all the given courses can be taken, while satisfying all the prerequisites. If there exists more than one such order, any one of them would be a correct answer. If no such order exists, return a special value: [-1].
Input: n=4, prerequisites=[[1, 0], [2, 0], [3, 1], [3, 2]]
Output: [0, 1, 2, 3]
According to the input:
● Course 0 must be done before both 1 and 2
● Courses 1 and 2 must be done before course 3
There are two orders in which one can take all four courses: [0, 1, 2, 3] and [0, 2, 1, 3]. Both are correct answers.
Function accepts two arguments: The number of courses n and the list of prerequisites.
Prerequisites are given in the form of a two-dimensional array (or a list of lists) of integers. Each inner array has exactly two elements — it is essentially a list of pairs. Each pair [X, Y] represents one prerequisite: Course Y must be completed before X X depends on Y).
Function must return an array (list) of integers.
If all given courses can be taken while satisfying all given prerequisites, the returned array must contain a possible ordering (if more than one such ordering exists, any one must be returned). Otherwise, the function must return an array (list) with one element -1 in it.
This is a topological sorting problem. We’ll cover two efficient sample solutions: one uses DFS, while the second keeps track of the in-degree of the graph nodes. They have the same time and space complexity in the Big-O notation in terms of the number of courses and the number of prerequisites.
Both algorithms need a directed graph to work with. So, let us build one.
For example, if n=4 and prerequisites=[[1, 0], [2, 0], [3, 1], [3, 2]], the graph would look like this:
The correct answer to the problem will be a topological order of the nodes. The two sample solutions discussed below are essentially two different implementations of the topological sort algorithm.
Topological ordering doesn’t exist if the graph has a cycle. Both sample solutions detect cycles in their respective ways and return the special value in that case.
Topological sort implementation via DFS goes like this:
That’s what dfs_solution.cpp does.
It uses a standard DFS approach for detecting cycles: as we run DFS, if we encounter an edge from the “current” node to one that is visited but isn’t finished with, that means there is a cycle.
O(n + e), where n is the number of courses/nodes, and e is the number of prerequisites/edges.
Top-level DFS takes O(n + e) time. On top of that, we:
O(n + e)
The graph we build takes O(n + e) space, and the DFS stack also takes up to O(n + e) space.
O(n + e)
Input takes O(e) space, output takes O(n), and the auxiliary space used is O(n + e).
In-degree of a node in a directed graph is the number of edges that end in that node. This approach to topological sorting is based on the observation that any node with zero in-degree can be readily added to the topological ordering. The same applies to the courses and dependencies: we can always start taking courses from one with no dependencies.
So, we first add any node with zero in-degree to the “answer” list. Then we “erase” that node from the graph.
Note: While conceptually we remove a node and edges that start from it from the graph, in the actual sample solution, modification of the “edges” list happens to be unnecessary because of how “in_degree” and “nodes_with_zero_in_degree” lists are used.
As a result, any courses dependent on that “erased” node will not have that dependency. If that was their last dependency, they now have none — so zero in-degree — and can be added to the “answer” in the next round.
We repeat this (picking any node with zero in-degree, adding it to the “answer” list, removing it from the graph) until we run out of nodes with zero in-degree, and we are done. “in_degree” list is used to locate nodes with zero in-degree efficiently (saving us a loop over the remaining nodes to find one with zero in-degree any time we need one).
Cycle detection: If we run out of nodes with zero in-degree while some nodes still remain in the graph, it means that the remaining nodes all have incoming edges. If we visually imagine such a portion of a graph, we can see that it necessarily has a cycle.
O(n + e), where n is the number of courses/nodes, and e is the number of prerequisites/edges.
Building the graph from input data takes O(n + e) time.
The main part of the solution consists of a FOR loop inside of a WHILE loop. Body of the WHILE loop is executed n times at worst, while the body of the FOR loop is executed e times at worst (even though the FOR loop itself is “attempted” up to n times). So, O(n + e) is the total.
The intermediate data structures for representing the graph and for tracking the in-degree of nodes take O(n + e) space.
O(n + e).
Attend our free webinar to amp up your career and get the salary you deserve.