杀死进程-LeetCode-582
英文版
582. Kill Process
Given n processes, each process has a unique PID (process id) and its PPID (parent process id).
Each process only has one parent process, but may have one or more children processes. This is just like a tree structure. Only one process has PPID that is 0, which means this process has no parent process. All the PIDs will be distinct positive integers.
We use two list of integers to represent a list of processes, where the first list contains PID for each process and the second list contains the corresponding PPID.
Now given the two lists, and a PID representing a process you want to kill, return a list of PIDs of processes that will be killed in the end. You should assume that when a process is killed, all its children processes will be killed. No order is required for
the final answer.
Example 1:
Input: pid = [1, 3, 10, 5] ppid = [3, 0, 5, 3] kill = 5 Output: [5,10] Explanation: 3 / \ 1 5 / 10 Kill 5 will also kill 10.
Note:
- - The given kill id is guaranteed to be one of the given PIDs.
- - n >= 1.
中文版
给你n个进程,每个进程都有一个唯一的PID(process id)和PPID(parent process id)。每个进程最多只有一个父进程,但可能有多个子进程。如果一个进程的PPID为0,表示它没有父进程。如果一个进程被杀死,那么它的子进程也会被杀死。输入两个相同长度的整数链表,第一个链表是每个进程的PID,第二个链表是对应位置进程的PPID,再输入一个将被杀死的进程的PID,请输出所有将被杀死的进程的PID。
例如,输入的PID链表为[1, 3, 10, 5],PPID链表为[3, 0, 5, 3],将被杀死的进程的PID为5,那么最终被杀死的进程有两个,PID分别为5和10。这是因为PID为10的进程是PID为5的进程的子进程。
分析
由于每个进程最多只有一个父进程,但可能有多个子进程,因此进程之间的关系可以用一棵树来表示。父进程对应树中的父节点,而子进程对应树中的子节点。如果进程[1, 3, 10, 5]的父进程为[3, 0, 5, 3],那么这些进程可以构成如下的一棵树:如样例中的树结构
因此,解决这个问题的第一步是根据输入的PID和PPID两个链表构建出一个树。树是一种特殊的图(Graph)。图有两种常用的表示方法,一是基于邻接表,二是基于邻接矩阵。由于树是一类比较稀疏的图,一般用邻接表更为高效。
用邻接表来表示一棵树,我们可以把整棵树存到一个HashMap中。这个HashMap的Key为进程的PID,Value为对应进程的所有子进程的PID,也就是树中对应节点的所有子节点。下面是根据PID和PPID链表构建树的参考代码:
1 private Map<Integer, List<Integer>> buildTree(List<Integer> pid, List<Integer> ppid) { 2 Map<Integer, List<Integer>> processTree = new HashMap<>(); 3 Iterator<Integer> iterator1 = pid.iterator(); 4 Iterator<Integer> iterator2 = ppid.iterator(); 5 while (iterator1.hasNext() && iterator2.hasNext()) { 6 int p = iterator1.next(); 7 int pp = iterator2.next(); 8 if (!processTree.containsKey(pp)) { 9 processTree.put(pp, new LinkedList<>()); 10 } 11 processTree.get(pp).add(p); 12 } 13 return processTree; 14 }
由于杀死一个进程会杀死它的所有子进程,那么对应到树中,杀死某个节点对应的进程,也就会杀死以该节点为根节点的子树中所有节点对应的所有进程。因此,我们可以把问题转化为:如何遍历以某节点为根节点的子树。
虽然这棵由进程构成的树不一定是二叉树(因为节点的子节点数目可能大于2),但遍历算法大同小异,仍然可以按照广度优先或者深度优先的顺序遍历。下面分别介绍两种方法的代码。
深度优先解
树的深度优先搜索的代码有递归和非递归两种写法。通常基于递归的代码更为简洁,深度优先算法需要定义一个递归函数用来杀死某个进程,如下面的辅助函数kill。在函数kill中,每当杀死进程target,如果它有子进程,则递归调用函数kill去杀死它的每一个子进程child。
1 public List<Integer> killProcess(List<Integer> pid, List<Integer> ppid, int target){ 2 Map<Integer, List<Integer>> processTree = buildTree(pid, ppid); 3 List<Integer> result = new LinkedList<>(); 4 kill(processTree, target, result); 5 return result; 6 } 7 8 private void kill(Map<Integer, List<Integer>> processTree, int target, List<Integer> result) { 9 result.add(target); 10 if (!processTree.containsKey(target)) { 11 return ; 12 } 13 for (int child : processTree.get(target)) { 14 kill(processTree, child, result); 15 } 16 }
也可以不构建树结构直接,利用这两个数组进行递归杀死进程。
1 public List<Integer> killProcess(List<Integer> pid, List<Integer> ppid, int kill) { 2 ArrayList<Integer> ans = new ArrayList<Integer>(); 3 ans.add(kill); 4 for (int i = 0; i < ppid.size(); i++) { 5 if (ppid.get(i) == kill) { 6 ans.addAll(killProcess(pid, ppid, pid.get(i))); 7 } 8 } 9 return ans; 10 }
广度优先解
基于广度优先的代码通常会用到队列(queue)。我们首先把第一个待杀死的进程的PID放入队列中,接下来只要队列中还有待杀死的进程,就重复执行如下步骤:首先从队列中去除一个进程,杀死它(即添加到输出的链表中);接着,如果该进程有子进程则把子进程添加到队列中。
1 public List<Integer> killProcess(List<Integer> pid, List<Integer> ppid, int target) { 2 Map<Integer, List<Integer>> processTree = buildTree(pid, ppid)); 3 List<Integer> result = new LinkedList<>(); 4 Queue<Integer> queue = new LinkedList<>(); 5 queue.add(target); 6 while (!queue.isEmpty()) { 7 int process = queue.remove(); 8 result.add(process); 9 if (processTree.containsKey(process)) { 10 for (int child : processTree.get(process)) { 11 queue.add(child); 12 } 13 } 14 } 15 return result; 16 }