英文版

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 = [, , , ]
ppid = [, , , ]
kill =
Output: [,]
Explanation: / \ / Kill will also kill .

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链表构建树的参考代码:

 private Map<Integer, List<Integer>> buildTree(List<Integer> pid, List<Integer> ppid) {
Map<Integer, List<Integer>> processTree = new HashMap<>();
Iterator<Integer> iterator1 = pid.iterator();
Iterator<Integer> iterator2 = ppid.iterator();
while (iterator1.hasNext() && iterator2.hasNext()) {
int p = iterator1.next();
int pp = iterator2.next();
if (!processTree.containsKey(pp)) {
processTree.put(pp, new LinkedList<>());
}
processTree.get(pp).add(p);
}
return processTree;
}

由于杀死一个进程会杀死它的所有子进程,那么对应到树中,杀死某个节点对应的进程,也就会杀死以该节点为根节点的子树中所有节点对应的所有进程。因此,我们可以把问题转化为:如何遍历以某节点为根节点的子树。

虽然这棵由进程构成的树不一定是二叉树(因为节点的子节点数目可能大于2),但遍历算法大同小异,仍然可以按照广度优先或者深度优先的顺序遍历。下面分别介绍两种方法的代码。

深度优先解

树的深度优先搜索的代码有递归和非递归两种写法。通常基于递归的代码更为简洁,深度优先算法需要定义一个递归函数用来杀死某个进程,如下面的辅助函数kill。在函数kill中,每当杀死进程target,如果它有子进程,则递归调用函数kill去杀死它的每一个子进程child。

 public List<Integer> killProcess(List<Integer> pid, List<Integer> ppid, int target){
Map<Integer, List<Integer>> processTree = buildTree(pid, ppid);
List<Integer> result = new LinkedList<>();
kill(processTree, target, result);
return result;
} private void kill(Map<Integer, List<Integer>> processTree, int target, List<Integer> result) {
result.add(target);
if (!processTree.containsKey(target)) {
return ;
}
for (int child : processTree.get(target)) {
kill(processTree, child, result);
}
}

也可以不构建树结构直接,利用这两个数组进行递归杀死进程。

 public List<Integer> killProcess(List<Integer> pid, List<Integer> ppid, int kill) {
ArrayList<Integer> ans = new ArrayList<Integer>();
ans.add(kill);
for (int i = 0; i < ppid.size(); i++) {
if (ppid.get(i) == kill) {
ans.addAll(killProcess(pid, ppid, pid.get(i)));
}
}
return ans;
}

广度优先解

基于广度优先的代码通常会用到队列(queue)。我们首先把第一个待杀死的进程的PID放入队列中,接下来只要队列中还有待杀死的进程,就重复执行如下步骤:首先从队列中去除一个进程,杀死它(即添加到输出的链表中);接着,如果该进程有子进程则把子进程添加到队列中。

 public List<Integer> killProcess(List<Integer> pid, List<Integer> ppid, int target) {
Map<Integer, List<Integer>> processTree = buildTree(pid, ppid));
List<Integer> result = new LinkedList<>();
Queue<Integer> queue = new LinkedList<>();
queue.add(target);
while (!queue.isEmpty()) {
int process = queue.remove();
result.add(process);
if (processTree.containsKey(process)) {
for (int child : processTree.get(process)) {
queue.add(child);
}
}
}
return result;
}

杀死进程-LeetCode-582的更多相关文章

  1. 奇安信集团笔试题:二叉树的最近公共祖先(leetcode236),杀死进程(leetcode582)

    1. 二叉树最近公共祖先     奇安信集团 2020校招 服务端开发-应用开发方向在线考试 编程题|20分2/2 寻祖问宗 时间限制:C/C++语言 1000MS:其他语言 3000MS 内存限制: ...

  2. sql杀死进程

    查询SQL所有的链接  并可以查看连接当前正在做什么操作..使用的什么语句.. SELECT spid, blocked, DB_NAME(sp.dbid) AS DBName, program_na ...

  3. Window通过cmd查看端口占用、相应进程、杀死进程

       在windows下启动程序时候经常出现端口占用, 修改本程序端口是一种解决思路,但是更多时候希望直接杀掉占用端口的程序: 一. 查看所有进程占用的端口 在开始-运行-cmd,输入: netsta ...

  4. Linux Shell脚本实现根据进程名杀死进程

    Shell脚本源码如下: #!/bin/sh #根据进程名杀死进程 if [ $# -lt 1 ] then echo "缺少参数:procedure_name" exit 1 f ...

  5. Linux如何查看进程、杀死进程、启动进程等常用命令

    Linux如何查看进程.杀死进程.启动进程等常用命令 关键字: linux 查进程.杀进程.起进程1.查进程    ps命令查找与进程相关的PID号:    ps a 显示现行终端机下的所有程序,包括 ...

  6. Window 通过cmd查看端口占用、相应进程、杀死进程等的命令【转】

    一. 查看所有进程占用的端口  在开始-运行-cmd,输入:netstat –ano可以查看所有进程 二.查看占用指定端口的程序  当你在用tomcat发布程序时,经常会遇到端口被占用的情况,我们想知 ...

  7. windows下根据端口号杀死进程

    Windows不像Linux,Unix那样,ps -ef 查出端口和进程号,然后根据进程号直接kill进程. Windows根据端口号杀死进程要分三步: 第一步 根据端口号寻找进程号 C:\>n ...

  8. linux 杀死进程的方法

    # kill -pid 注释:标准的kill命令通常都能达到目的.终止有问题的进程,并把进程的资源释放给系统.然而,如果进程启动了子进程,只杀死父进程,子进程仍在运行,因此仍消耗资源.为了防止这些所谓 ...

  9. centos(linux) 下如何查看端口占用情况及杀死进程

    使用这个命令:netstat -nap [root@Jaosn sphinx]# netstat -nap Active Internet connections (servers and estab ...

随机推荐

  1. Apache版本hadoop-2.6.0.tar.gz平台下搭建Hue

    不多说,直接上干货! http://archive.apache.org/dist/ http://www.cnblogs.com/smartloli/p/4527168.html http://ww ...

  2. visual studio 2008 快捷键

    Ctrl+m+Crtr+o折叠所有大纲Ctrl+M+Crtr+P: 停止大纲显示Ctrl+K+Crtr+C: 注释选定内容Ctrl+K+Crtr+U: 取消选定注释内容Ctrl+J : 列出成员 智能 ...

  3. tomcat启动(Ⅷ)--请求最终目的地 getContainer().getPipeline().getFirst().invoke(request, response)

    当tomcat的Conector保存着StandardService实例,而StandardService保存着Container的实例 当Http11NioProcessor.process()方法 ...

  4. InterView之PHP

    PHP HTTP Keep-Alive的作用 作用 Keep-Alive:使客户端到服务器端的连接持续有效,当出现对服务器的后继请求时,Keep-Alive功能避免了建立或者重新建立连接.Web服务器 ...

  5. Eth 部署智能合约

    首先要开发以太坊的智能合约,需要EVM(以太坊虚拟机),也就是需要运行的环境,我们可以通过 geth 来设置开发环境: geth --datadir testNet --dev console 2&g ...

  6. java-多线程future等待返回

    多线程中需要返回值, java中有个fork/join模型, 没有细研究, 简单实用callable进行了返回 Thread1 package com.iwhere.easy.travel.test. ...

  7. equal&==&hashcode

    == 和 equals 的区别 Object类中的equals方法和“==”是一样的,没有区别,而String类,Integer类等等一些类,是重写了equals方法,才使得equals和“==不同” ...

  8. SqlDataReader的关闭问题,报错:“阅读器关闭时尝试调用 Read 无效”

    SqlDataReader dr = cmd.ExecuteReader(CommandBehavior.CloseConnection);//关闭SqlDataReader 会自动关闭Sqlconn ...

  9. Shell脚本编写5-----Shell 基本运算符

    算术运算符 原生bash不支持简单的数学运算,但是可以通过其他命令来实现,例如 awk 和 expr,expr 最常用.expr 是一款表达式计算工具,使用它能完成表达式的求值操作.例如: 两个数相加 ...

  10. HA_Snapshots 数据库快照

    1. 创建测试数据库HA_Snapshot 2. 创建快照 create database HA_Snapshot_20 on ( name = HA_Snapshot, filename = '.. ...