需解决的问题:

   想开发一个JMX的客户端来配置一个本地Java程序。具体要求如下:


请不要推荐使用JConsole!JConsole不合适,因为JConsole是一个通用的JMX客户端,对主程序的性能存在负面影响。

Oracle网站上有一个使用RMIConnector和“主机 : 端口号”为参数的例子,但是我不知道在什么地方设置JMX的端口号?

JConsole可以选择PID来连接Java进程。但是在JMX的API中,我找不到任何方法是使用PID作为参数的。

解答:

我们使用一些类似下面的程序来连接JMX服务器。不过,需要在运行服务器时指定以下参数:


-Dcom.sun.management.jmxremote

-Dcom.sun.management.jmxremote.authenticate=false

-Dcom.sun.management.jmxremote.port=1234

-Dcom.sun.management.jmxremote.ssl=false

为了绑定特定地址,你需要在增加下面的VM参数:


1-Djava.rmi.server.hostname=A.B.C.D

这样,你就可以像下面的JMX客户端代码一样连接你的服务器了:



String host = "localhost";  // or some A.B.C.D

int port = 1234;

String url = "service:jmx:rmi:///jndi/rmi://" + host + ":" + port + "/jmxrmi";

JMXServiceURL serviceUrl = new JMXServiceURL(url);

JMXConnector jmxConnector = JMXConnectorFactory.connect(serviceUrl, null);

try {

   MBeanServerConnection mbeanConn = jmxConnector.getMBeanServerConnection();

   // now query to get the beans or whatever

   Set<ObjectName> beanSet = mbeanConn.queryNames(null, null);

   ...

} finally {

   jmxConnector.close();

}

我们也可以不使用VM参数,代码本身可以通过编程在指定端口号发布。但是我想这个已经比你的需求更复杂了。


如果要根据PID来连接,据我现在所知,你需要使用Java 6以上的版本。我没使用过下面的代码,但是看起是可以正常工作的:



List<VirtualMachineDescriptor> vms = VirtualMachine.list();

for (VirtualMachineDescriptor desc : vms) {

    VirtualMachine vm;

    try {

        vm = VirtualMachine.attach(desc);

    } catch (AttachNotSupportedException e) {

        continue;

    }

    Properties props = vm.getAgentProperties();

    String connectorAddress =

        props.getProperty("com.sun.management.jmxremote.localConnectorAddress");

    if (connectorAddress == null) {

        continue;

    }

    JMXServiceURL url = new JMXServiceURL(connectorAddress);

    JMXConnector connector = JMXConnectorFactory.connect(url);

    try {

        MBeanServerConnection mbeanConn = connector.getMBeanServerConnection();

        Set<ObjectName> beanSet = mbeanConn.queryNames(null, null);

        ...

    } finally {

        jmxConnector.close();

    }

}

我已发布过一个新的SimpleJMX包,该包能帮助很简单的启动一个JMX服务,并向远程客户端发送beans。


//创建一个新的服务器并监听8000端口

JmxServer jmxServer = new JmxServer(8000);

//启动服务器

jmxServer.start();

//注册下面定义的lookupCache对象

jmxServer.register(lookupCache);

jmxServer.register(someOtherObject);

//停止服务

jmxServer.stop();

该包确实有一个客户端的接口,但是当前没有人一种机制是可以通过PID来查找进程的,只支持主机/端口的组合方式查找。


PS:如有什么问题,请直接在群457036818提出。