将 Jolokia 代理与 RMI 服务器一起使用

Using Jolokia agent with an RMI-Server

我在将 Jolokia 与 RMI 服务结合使用时遇到问题。一旦 RMI 服务启动,就无法通过 http 访问 Jolokia。

我创建了一个示例 class 来重现问题:

package com.example.rmi;

import java.net.InetAddress;
import java.rmi.Naming;
import java.rmi.RemoteException;
import java.rmi.registry.LocateRegistry;
import java.rmi.server.UnicastRemoteObject;

public class RMITest {

    private class RMIService extends UnicastRemoteObject {

        private static final long serialVersionUID = 1L;

        protected RMIService() throws RemoteException {
            super();
        }

        public void doStuff() {
            System.out.println("Processing...");

        }

    }

    public static void main(String[] args) throws RemoteException {
        RMITest rmiServer = new RMITest();
        rmiServer.init();
    }

    private void init() throws RemoteException {
        RMIService rmiService = new RMIService();

        try {
            System.out.println("Starting RMI-Service...");
            String hostname = InetAddress.getLocalHost().getHostName();
            System.setProperty("java.rmi.server.hostname", hostname);

            int port = 2005;
            LocateRegistry.createRegistry(port);
            Naming.rebind("rmi://" + hostname + ":" + port
                    + "/RMIService", rmiService);
            System.out.println("RMI-Service started!");

        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

如果我将主要方法更改为不启动 RMI 服务,则可以再次通过 http URL http://127.0.0.1:8778/jolokia/ 访问 Jolokia:

public static void main(String[] args) throws RemoteException {
    RMITest rmiServer = new RMITest();
    //rmiServer.init();

    while(true) {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
   }

该服务是一个可运行的 jar。这是我用来启动应用程序的命令:

java -javaagent:jolokia-jvm-1.3.7-agent.jar=port=8778,host=localhost -jar RMITest-0.0.1-SNAPSHOT.jar

我从官网下载了jolokia代理:Jolokia Agent Download

我找到了一个解决方法,即在启动 RMI 服务后以编程方式启动代理。

在下面的Whosebug-Post中描述了如何做:Starting a Java agent after program start

所以我在 运行 初始化方法之后启动以下静态方法:

public static void attachGivenAgentToThisVM(String pathToAgentJar) {
    try {
        String nameOfRunningVM = ManagementFactory.getRuntimeMXBean().getName();
        String pid = nameOfRunningVM.substring(0, nameOfRunningVM.indexOf('@'));
        VirtualMachine vm = VirtualMachine.attach(pid);
        vm.loadAgent(pathToAgentJar, "");
        vm.detach();
    } catch (Exception e) {
        e.printStackTrace();
    }
}

还需要依赖"tools.jar":

<dependency>
    <groupId>com.sun</groupId>
    <artifactId>tools</artifactId>
    <version>1.8</version>
    <scope>system</scope>
    <systemPath>${java.home}/../lib/tools.jar</systemPath>
</dependency>

我还是更喜欢直接从命令行启动代理。因此,非常感谢更好的解决方案。