There are ‘N’ tasks, labeled from ‘0’ to ‘N-1’. Each task can have some prerequisite tasks which need to be completed before it can be scheduled. Given the number of tasks and a list of prerequisite pairs, write a method to find the ordering of tasks we should pick to finish all tasks.
Example 1:
Input:
Tasks=3,
Prerequisites=[0, 1], [1, 2]
Output: [0, 1, 2]
Explanation: To execute task '1', task '0' needs to finish first.
Similarly, task '1' needs to finish before '2' can be scheduled.
A possible scheduling of tasks is: [0, 1, 2]
Example 2:
Input:
Tasks=3,
Prerequisites=[0, 1], [1, 2], [2, 0]
Output: []
Explanation: The tasks have cyclic dependency,
therefore they cannot be scheduled.
Example 3:
Input:
Tasks=6,
Prerequisites=[2, 5], [0, 5], [0, 4], [1, 4], [3, 2], [1, 3]
Output: [0 1 4 3 2 5]
Explanation: A possible scheduling of tasks is: [0 1 4 3 2 5]
Solution
publicstaticbooleanisSchedulingPossible(int tasks,int[][] prerequisites) {List<Integer> res =newArrayList<>();// 1. Initialize the graphMap<Integer,List<Integer>> graph =newHashMap<>();Map<Integer,Integer> indegree =newHashMap<>();for (int i =0; i < tasks; i++) {graph.put(i,newArrayList<>());indegree.put(i,0); }// 2. Build the graphfor (int i =0; i <prerequisites.length; i++) {int start = prerequisites[i][0], end = prerequisites[i][1];graph.get(start).add(end);indegree.put(end,indegree.get(end) +1); }// 3. Add all the sources(i.e., vertices with in-degree 0)Queue<Integer> sources =newLinkedList<>();for (Map.Entry<Integer,Integer> entry :indegree.entrySet()) {if (entry.getValue() ==0)sources.offer(entry.getKey()); }// 4. Process the sources and add it to the result, decrement the in-degree by// one of all the childrenwhile (!sources.isEmpty()) {int vertex =sources.poll();res.add(vertex);for (int child :graph.get(vertex)) {indegree.put(child,indegree.get(child) -1);if (indegree.get(child) ==0)sources.add(child); } }// if res doesn't contain all tasks, there is a cyclic dependency between tasks, therefore, we// will not be able to schedule all tasksif (res.size() != tasks)returnnewArrayList<>();return res; }
Time complexity
In step ‘4’, each task can become a source only once and each edge (prerequisite) will be accessed and removed once. Therefore, the time complexity of the above algorithm will be O(V+E), where ‘V’ is the total number of tasks and ‘E’ is the total number of prerequisites.
Space complexity
The space complexity will be O(V+E), since we are storing all of the prerequisites for each task in an adjacency list.