如何用JMX連接本地JVM上運行的Java程序


問題:

我需要使用JMX連接本地JVM的Java程序。換句話說,我想開發一個JMX的客戶端來配置一個本地Java程序。

  • 請不要推薦使用JConsole!JConsole不合適,因為JConsole是一個通用的JMX客戶端,對主程序的性能存在負面影響。
  • Oracle網站上有一個使用RMIConnector和“主機 : 端口號”為參數的例子,但是我不知道在什么地方設置JMX的端口號?
  • JConsole可以選擇PID來連接Java進程。但是在JMX的API中,我找不到任何方法是使用PID作為參數的。

回復:

我們使用一些類似下面的程序來連接JMX服務器。不過,需要在運行服務器時指定以下參數:

1
2
3
4
-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客戶端代碼一樣連接你的服務器了:

1
2
3
4
5
6
7
8
9
10
11
12
13
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以上的版本。我沒使用過下面的代碼,但是看起是可以正常工作的:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
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。

1
2
3
4
5
6
7
8
9
//創建一個新的服務器並監聽8000端口
JmxServer jmxServer = new JmxServer( 8000 );
//啟動服務器
jmxServer.start();
//注冊下面定義的lookupCache對象
jmxServer.register(lookupCache);
jmxServer.register(someOtherObject);
//停止服務
jmxServer.stop();

該包確實有一個客戶端的接口,但是當前沒有人一種機制是可以通過PID來查找進程的,只支持主機/端口的組合方式查找。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM