Java获取当前进程ID以及所有Java进程的进程ID

2018-01-18 10:16:33

首先是获取当前Java运行的Java进程ID,这个是网上常见的,也就是Java程序自身将进程ID打印出来:

package com.test;import java.lang.management.ManagementFactory;import java.lang.management.RuntimeMXBean;public class Target {

    public static void main(String[] args) throws InterruptedException {
        System.out.println(getProcessID());        while(true) {
            Thread.sleep(10000);
        }
    }    public static final int getProcessID() {  
        RuntimeMXBean runtimeMXBean = ManagementFactory.getRuntimeMXBean();
        System.out.println(runtimeMXBean.getName());        return Integer.valueOf(runtimeMXBean.getName().split("@")[0])  
                .intValue();  
    } 

}1234567891011121314151617181920212223

运行结果:

2896@PC-20150603VRPL289612

ManagementFactory是一个在运行时管理和监控Java VM的工厂类,它能提供很多管理VM的静态接口,比如RuntimeMXBean

RuntimeMXBean是Java虚拟机的运行时管理接口.

当前进程ID为2896。

但这种让Java进程自行打印ID的方式并不太好用,因为很多时候我们是需要在不修改代码另一个Java程序代码的情况下知道这个正在运行着的Java进程的ID,如果你也有这个需求,可以看一下下面的解决方法:

获取所有正在运行着的Java进程

package com.test;import java.util.HashSet;import java.util.Set;import sun.jvmstat.monitor.MonitoredHost;import sun.jvmstat.monitor.MonitoredVm;import sun.jvmstat.monitor.MonitoredVmUtil;import sun.jvmstat.monitor.VmIdentifier;public class ProcessID {

    public static void main(String[] args) throws Exception {         // 获取监控主机
         MonitoredHost local = MonitoredHost.getMonitoredHost("localhost");         // 取得所有在活动的虚拟机集合
         Set<?> vmlist = new HashSet<Object>(local.activeVms());         // 遍历集合,输出PID和进程名
         for(Object process : vmlist) {
             MonitoredVm vm = local.getMonitoredVm(new VmIdentifier("//" + process));             // 获取类名
             String processname = MonitoredVmUtil.mainClass(vm, true);
             System.out.println(process + " ------> " + processname);
         }
     }
}123456789101112131415161718192021222324252627

运行结果:

2752 ------> 
5172 ------> com.test.Target5308 ------> com.test.ProcessID123

这样,就可以根据类名找到进程号了。

MonitoredHost等类位于${JAVA_HOME}/lib/tools.jar

下面是一个可以直接根据类找到对应Java进程ID的方法:

package com.test;import java.net.URISyntaxException;import java.util.HashSet;import java.util.Set;import sun.jvmstat.monitor.MonitorException;import sun.jvmstat.monitor.MonitoredHost;import sun.jvmstat.monitor.MonitoredVm;import sun.jvmstat.monitor.MonitoredVmUtil;import sun.jvmstat.monitor.VmIdentifier;public class ProcessID {

    public static void main(String[] args) throws Exception {         int pid = getProcess(Target.class);
         System.out.println("PID: "+pid);
    }    public static int getProcess(Class<?> cls) throws MonitorException, URISyntaxException {        if(cls == null) {            return -1;
        }        // 获取监控主机
        MonitoredHost local = MonitoredHost.getMonitoredHost("localhost");        // 取得所有在活动的虚拟机集合
        Set<?> vmlist = new HashSet<Object>(local.activeVms());        // 遍历集合,输出PID和进程名
        for(Object process : vmlist) {
            MonitoredVm vm = local.getMonitoredVm(new VmIdentifier("//" + process));            // 获取类名
            String processname = MonitoredVmUtil.mainClass(vm, true);            if(cls.getName().equals(processname)) {                return ((Integer)process).intValue();
            }
        }        return -1;
    }
}1234567891011121314151617181920212223242526272829303132333435363738394041

运行结果:

PID: 51721

源码位于我的Github项目aoptracer的com.test包里


  • 2017-07-06 10:02:18

    大白话讲解Promise(一)

    去年6月份, ES2015正式发布(也就是ES6,ES6是它的乳名),其中Promise被列为正式规范。作为ES6中最重要的特性之一,我们有必要掌握并理解透彻。本文将由浅到深,讲解Promise的基本概念与使用方法。

  • 2017-07-11 21:54:14

    MYSQL5.7版本sql_mode=only_full_group_by问题

    一旦开启 only_full_group_by ,感觉,group by 将变成和 distinct 一样,只能获取受到其影响的字段信息,无法和其他未受其影响的字段共存,这样,group by 的功能将变得十分狭窄了

  • 2017-07-14 13:51:58

    NodeJS连接MySQL出现Cannot enqueue Handshake after invoking quit.

    原因在于node连接上mysql后如果因网络原因丢失连接或者用户手工关闭连接后,原有的连接挂掉,需要重新连接;如下代码,每次访问结束都关闭,每次开始访问前重连接下,代码中没有监听连接的fatal错误,copy需谨慎

  • 2017-07-14 13:53:02

    nodejs解决mysql和连接池(pool)自动断开问题

    最近在做一个个人项目,数据库尝试使用了mongodb、sqlite和mysql。分享一下关于mysql的连接池用法。项目部署于appfog,项目中我使用连接池链接数据库,本地测试一切正常。上线以后,经过几次请求两个数据接口总是报503。一直不明就里,今天经过一番排查终于顺利解决了。

  • 2017-07-15 16:13:26

    设置MySQL里的wait_timeout

    如果你没有修改过MySQL的配置,缺省情况下,wait_timeout的初始值是28800。

  • 2017-07-16 20:13:14

    nodejs,express 自制错误日志

    对于同步执行的代码,以上的处理已经足够简单。然而,当异步程序在执行时抛出异常的情况,Express 就无能为力。原因在于当你的程序开始执行回调函数时,它原来的栈信息已经丢失。