spring pojo_使用Spring将POJO公开为JMX MBean

spring pojo

这是一个非常不错的教程,介绍了如何通过我们最新的JCG合作伙伴The Holy Java ”博客(很酷的名字)实现“ 用Spring轻松将POJO作为JMX MBean公开 ”。

(注意:对原始帖子进行了少量编辑以提高可读性)

Java管理扩展(JMX)技术是一种检查或更改变量状态或通过(例如) JConsole之类的管理GUI在(远程)运行的应用程序中调用方法的好方法。 Spring使在几分钟之内仅用很少的配置就可以将任何POJO公开为JMX MBean变得微不足道。 Spring JMX文档非常好,但是有一段时间我一直在挣扎,因此想在这里记录正确的解决方案。

我需要在服务器上运行的IBM JVM 1.5上使用Spring 2.5监视命令行Java应用程序。 监视将在Sun JVM 1.6上使用jconsole作为PC上的JMX客户端执行。 以下所有XML代码段均来自相应的Spring application-context.xml。

将POJO变成MBean

JMX可以公开以原语或复杂数据类型为参数的getter,setter和操作(尽管除少数特殊类型外,其他类型都要求客户端具有类)。 您告诉Spring将POJO公开为MBean,如下所示:

<bean id="myMBean"
        class="my.package.JobPerformanceStats"
        factory-method="instance" />

<bean class="org.springframework.jmx.export.MBeanExporter" lazy-init="false">
        <property name="beans">
          <map>
                <entry key="bean:name=MyMBeanName" value-ref="myMBean"/>
          </map>
        </property>
</bean>

首先,您声明POJO类的实例– myMBean(出于其他原因,我使用老式的单例,并使用JobPerformanceStats.instance()访问Bean)。 接下来,使用lazy-init =“ false”声明一个MBeanExporter并向其介绍您的bean。 (还有其他方法可以执行此操作,包括自动发现。)然后,该bean将在其键下可见,即“ bean:name = MyMBeanName”,JConsole将其显示为“ MyMBeanName”。

注意,由于MBeanExporter使用新的java.lang.management包,因此它仅在JVM 1.5+下工作。 在JDK 1.4下,Spring会因以下错误而失败:

java.lang.NoClassDefFoundError:javax / management / MBeanServerFactory
在org.springframework.jmx.support.MBeanServerFactoryBean.createMBeanServer

默认情况下,它将公开所有公共方法和属性。 您可以通过多种方式进行更改,例如在界面的帮助下。

如果不是在已经提供MBean服务器的容器中运行(在这里就是我的情况),则必须告诉Spring启动一个容器:

<bean class="org.springframework.jmx.support.MBeanServerFactoryBean"/>

启用远程访问

要使MBean可从另一台机器访问,必须通过声明配置有适当通信机制的ConnectorServerFactoryBean将其公开。

通过JMXMP进行远程访问

默认情况下, ConnectorServerFactoryBean通过JMX消息传递协议(JMXMP)使用地址公开MBean。

服务:jmx:jmxmp:// localhost:9875

<bean class="org.springframework.jmx.support.ConnectorServerFactoryBean" />

但是,现成的协议不支持此协议,因此必须在MBean应用程序和jconsole客户端的类路径上都包括OpenDMK的一部分jmxremote_optional.jar

org.springframework.beans.factory.BeanCreationException:创建名称为'org.springframework.jmx.support.ConnectorServerFactoryBean#0的bean时出错? 在类路径资源[application-context.xml]中定义:初始化方法的调用失败; 嵌套的异常是java.net.MalformedURLException:不支持的协议:jmxmp

通过RMI进行远程访问

或者,您可以通过RMI公开MBean,它没有其他依赖性:

<!--
Now expose the server for remote access via RMI
Local access:   service:jmx:rmi://localhost/jndi/rmi://localhost:10099/myconnector
Remote access:  service:jmx:rmi:///jndi/rmi://your.host:10099/myconnector
or service:jmx:rmi://localhost/jndi/rmi://localhost:10099/myconnector
-->
<bean
 class="org.springframework.jmx.support.ConnectorServerFactoryBean"
 depends-on="rmiRegistry">
 <property name="objectName" value="connector:name=rmi" />
 <property name="serviceUrl"
 value="service:jmx:rmi://localhost/jndi/rmi://localhost:10099/myconnector" />
</bean>

<bean id="rmiRegistry"
 class="org.springframework.remoting.rmi.RmiRegistryFactoryBean">
 <property name="port" value="10099" />
</bean>

但是,还必须避免一些陷阱:

1.必须启动RMI注册表,以便连接器可以在此处注册MBean。 它不会为你开始
2.您必须确保在连接器尝试使用之前启动注册表,方法是在连接器之前声明它,或者通过使用depends-on属性将该依赖关系明确化

如果未正确设置,则会出现如下异常:

org.springframework.beans.factory.BeanCreationException:创建名称为'org.springframework.jmx.support.ConnectorServerFactoryBean#0的bean时出错? 在类路径资源[application-context.xml]中定义:初始化方法的调用失败; 嵌套异常是java.io.IOException:无法绑定到URL [rmi:// localhost:10099 / jmxrmi]:javax.naming.ServiceUnavailableException [根本异常是java.rmi.ConnectException:连接被拒绝托管给主机:localhost; 嵌套的异常是:java.net.ConnectException:拒绝连接:connect]。

通过SSH隧道访问的本地MBean服务器

为了提高安全性,您可能希望不要通过仅从本地计算机(127.0.0.1)访问MBean并使用SSH隧道使MBean暴露给远程访问,以便远程JConsole可以将它们作为本地应用程序进行访问。 这当然是可能的,但可能会很困难,因为通常JMX会通过RMI进行访问,RMI 使用两个端口 :一个用于RMI Registry,另一个用于实际服务(此处为MBean服务器),通常在运行时随机选择,而您d需要同时隧穿。 幸运的是, Spring使配置两个端口成为可能

<bean
        class="org.springframework.jmx.support.ConnectorServerFactoryBean"
        depends-on="rmiRegistry">
        <property name="objectName" value="connector:name=rmi" />
        <property name="serviceUrl"
                value="service:jmx:rmi://127.0.0.1:STUBPORT/jndi/rmi://localhost:REGISTRYPORT/myconnector" />
</bean>

<bean id="rmiRegistry"
        class="org.springframework.remoting.rmi.RmiRegistryFactoryBean">
        <property name="port" value="REGISTRYPORT" />
</bean>

将STUBPORT和REGISTRYPORT替换为合适的编号,然后隧道传输这两个编号。 请注意,连接器的serviceUrl和RMI注册表的端口属性中的REGISTRYPORT号相同。
警告:上面的配置实际上并不能阻止从远程应用程序直接访问。 为了真正强制RMI注册表仅侦听来自本地主机的连接,我们可能需要在不带Spring的Sun JVM下设置系统属性com.sun.management.jmxremote。 另外,要强制注册表使用IP 120.0.0.1,我们需要设置java.rmi.server.hostname = localhost(也适用于Spring)。 请参阅有关强制本地访问的讨论 。 我不确定如何使用Spring获得相同的结果,同时仍然保留指定两个RMI端口的功能。 还要检查Spring RmiServiceExporter的JavaDoc。

相关文章和文档:

与Jconsole连接

启动JConsole并键入适当的远程地址,例如

服务:jmx:rmi:/// jndi / rmi://your.server.com:10099 / myconnector

如果连接到远程计算机上的应用程序,则可以通过RMI访问your.server.com。

关于连接URL,如果您有一个连接器,其serviceUrl为

服务:jmx:rmi:// myhost:9999 / jndi / rmi:// localhost:10099 / myconnector

然后,可以从客户使用

服务:jmx:rmi:// myhost:9999 / jndi / rmi://your.server.com:10099 / myconnector

或简单地

服务:jmx:rmi:/// jndi / rmi://your.server.com:10099 / myconnector

因为,根据JMX 1.2远程API规范(第90页):

…主机名和端口号
#(示例中为myhost:9999)不被客户端使用,如果
#目前,基本上是评论。 连接器服务器地址
#实际上存储在序列化的存根(/ stub /形式)或
#目录条目(/ jndi /格式)。

IBM JVM,JConsole和JMX配置

IBM JVM 5 SDK指南指出IBM SDK也包含JConsole并识别与JMX相关的相同系统属性 ,即com.sun.management.jmxremote。*(尽管未提及“ com.sun.management.jmxremote”本身)。 )。
请注意,IBM JConsole有所不同,例如,它缺少“本地”选项卡,该选项卡由指定命令行选项connection = localhost代替(在SDK指南中搜索“ JConsole监视工具的本地选项卡”)。

进一步改进

JVM 1.5:公开MemoryMXBean

从Java 5.0开始,有几个有用的平台MBean提供有关JVM的信息,包括java.lang.management.MemoryMXBean,它使您可以查看堆使用情况,调用GC等。

您可以按如下所示将其提供给JConsole和其他JMX代理使用(尽管必须有更简单的方法):

<bean class="org.springframework.jmx.export.MBeanExporter" lazy-init="false">
        <property name="beans">
          <map>
                <entry key="bean:name=Memory2" value-ref="memProxy"/>
                <!-- other exported beans may follow ... -->
          </map>
        </property>
</bean>

<bean id="memProxy"
        class="java.lang.management.ManagementFactory"
        factory-method="getMemoryMXBean"
        />

更新:通过使用工厂方法getPlatformMBeanServer将Spring的MBeanServerFactoryBean替换为java.lang.management.ManagementFactory,似乎确实存在直接暴露平台MBean更好的方法。 当然,这需要JVM 1.5+。

通过密码验证提高安全性

通过RMI访问MBean可能受密码保护。 根据讨论, 在服务器连接器上配置了身份验证

<bean
        class="org.springframework.jmx.support.ConnectorServerFactoryBean"
        depends-on="rmiRegistry">
        <property name="objectName" value="connector:name=rmi" />
        <property name="serviceUrl"
                value="service:jmx:rmi://localhost/jndi/rmi://localhost:10099/myconnector" />
    <property name="environment">
    <!-- the following is only valid when the sun jmx implementation is used -->
        <map>
            <entry key="jmx.remote.x.password.file" value="etc/security/jmxremote.password"/>
            <entry key="jmx.remote.x.access.file" value="etc/security/jmxremote.access"/>
        </map>
    </property>
</bean>

passwd和access文件遵循可在JDK / jre / lib / management文件夹中找到的模板。

摘要

使用Spring将POJO作为MBean公开很容易,只是不要忘记启动MBean服务器和连接器。 对于JMXMP,请包括jmxmp_impl。 对于RMI,请确保在类路径上添加jar,并确保在连接器之前启动RMI注册表。

相关文章:


翻译自: https://www.javacodegeeks.com/2011/02/expose-pojo-jmx-mbean-spring.html

spring pojo

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值