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

jopen 9年前發布 | 21K 次閱讀 JMX Java開發

問題:

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

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

    回復:

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

    -Dcom.sun.management.jmxremote
    -Dcom.sun.management.jmxremote.authenticate=false
    -Dcom.sun.management.jmxremote.port=1234
    -Dcom.sun.management.jmxremote.ssl=false

    為了綁定特定地址,你需要在增加下面的VM參數:

    -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來查找進程的,只支持主機/端口的組合方式查找。

    原文鏈接: stackoverflow 翻譯: ImportNew.com - paddx
    譯文鏈接: http://www.importnew.com/15982.html

 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!