4

如果在启动 JVM 时设置了“com.sun.management.jmxremote”系统属性,则可以运行 jconsole 或 visualvm 并连接到该本地 mBeanServer。我想做他们正在做的同样的事情,但不知道怎么做。

是否有可用于识别本地运行的 JVM 的服务 URL?

我知道我可以通过在特定端口上设置 jmxmp 或 rmi 侦听器然后连接到该端口来做到这一点,但我不想这样做,因为这意味着我必须管理端口并知道哪个端口连接到哪个端口jvm(我们在同一台服务器上运行多个 jvm)。

4

2 回答 2

7

感谢 Nicholas 和 Scott 的发帖和讨论。能够根据该信息和一些试验和错误将其拼凑在一起。

分享让其他人不必做同样的腿部工作。这将附加到 JVM 并获取其内存使用情况。

import com.sun.tools.attach.VirtualMachine;
import com.sun.tools.attach.VirtualMachineDescriptor;
import com.sun.tools.attach.spi.AttachProvider;

import javax.management.MBeanServerConnection;
import javax.management.ObjectName;
import javax.management.openmbean.CompositeData;
import javax.management.remote.JMXConnector;
import javax.management.remote.JMXConnectorFactory;
import javax.management.remote.JMXServiceURL;

public class AttachFun {

    public static void main(String[] args) throws Exception {
        final AttachProvider attachProvider = AttachProvider.providers().get(0);

        VirtualMachineDescriptor descriptor = null;
        for (VirtualMachineDescriptor virtualMachineDescriptor : attachProvider.listVirtualMachines()) {
            if (pickThisOne(virtualMachineDescriptor)) {
                descriptor = virtualMachineDescriptor;
                break;
            }
        }

        if (descriptor == null) throw new RuntimeException("You didn't pick one");

        final VirtualMachine virtualMachine = attachProvider.attachVirtualMachine(descriptor);
        virtualMachine.loadAgent("/System/Library/Frameworks/JavaVM.framework/Home/lib/management-agent.jar", "com.sun.management.jmxremote");
        final Object portObject = virtualMachine.getAgentProperties().get("com.sun.management.jmxremote.localConnectorAddress");

        final JMXServiceURL target = new JMXServiceURL(portObject + "");
        final JMXConnector connector = JMXConnectorFactory.connect(target);
        final MBeanServerConnection remote = connector.getMBeanServerConnection();

        final ObjectName memory = new ObjectName("java.lang:type=Memory");
        CompositeData cd = (CompositeData) remote.getAttribute(memory, "HeapMemoryUsage");

        final Long used = (Long) cd.get("used");
        System.out.println(used);

    }

    private static boolean pickThisOne(VirtualMachineDescriptor virtualMachineDescriptor) {
        // TODO
        return false;
    }
}
于 2012-10-29T06:59:22.573 回答
1

您需要附加 API。它将在同一主机上找到正在运行的 [支持 API 的] JVM,并在每个主机上返回元数据,并启用 JMXConnector。这基本上就是您在启动 JConsole 时看到的内容,并且连接菜单显示正在运行的 JVM,包括未使用任何特殊 JMX 命令行指令启动的 Java 1.6+ 实例。

//尼古拉斯

于 2010-12-31T10:39:19.373 回答