Spring與JMX集成

Spring與JMX集成,實現方式靈活而且簡單,主要體現在:

1、可以自動探測實現MBean接口的MBean對象,而且可以將一個普通的Spring Bean註冊爲MBean;

2、定製管理MBean的接口,根據需要暴露特定管理MBean的操作;

3、使用註解定義MBean管理接口;

4、可以實現對本地和遠程MBean的代理。

 

標準MBean管理

對於實現標準MBean接口MBean資源,在Spring中可以設置不同的探測模式,主要是通過MBeanExporter來實現。例如,定義MBean管理接口:

  1. package org.shirdrn.spring.jmx.autodetect;  
  2.   
  3. public interface MyObjectMBean {  
  4.     public long getId();  
  5.     public void setId(long id);  
  6.     public String getName();  
  7.     public void setName(String name);  
  8.     public String show();  
  9. }  

對應的MBean實現,如下所示:

  1. package org.shirdrn.spring.jmx.autodetect;  
  2.   
  3. import org.apache.log4j.Logger;  
  4.   
  5. public class MyObject implements MyObjectMBean {  
  6.       
  7.     private static final Logger LOG = Logger.getLogger(MyObject.class);  
  8.     private long id;  
  9.     private String name;  
  10.       
  11.     public MyObject() {  
  12.         super();  
  13.     }  
  14.       
  15.     public MyObject(long id, String name) {  
  16.         super();  
  17.         this.id = id;  
  18.         this.name = name;  
  19.     }  
  20.   
  21.     public long getId() {  
  22.         return id;  
  23.     }  
  24.   
  25.     public void setId(long id) {  
  26.         this.id = id;  
  27.     }  
  28.   
  29.     public String getName() {  
  30.         return name;  
  31.     }  
  32.   
  33.     public void setName(String name) {  
  34.         this.name = name;  
  35.     }  
  36.   
  37.     public String show() {  
  38.         StringBuffer sb = new StringBuffer().append("id=").append(id).append(  
  39.                 ", name=").append(name);  
  40.         LOG.info("show()=" + sb.toString());  
  41.         return sb.toString();  
  42.     }  
  43. }  

上面是一個標準MBean的實現,可以使用JavaSE 6平臺的JMX服務來管理,但是在Spring集成的環境下,也可以重用這些MBean實現,例如下面是一個典型的配置:

  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <beans  
  3.     xmlns="http://www.springframework.org/schema/beans"  
  4.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  5.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">  
  6.     <bean id="mbServer" class="org.springframework.jmx.export.MBeanExporter">  
  7.         <property name="autodetectModeName">  
  8.             <value>AUTODETECT_ALL</value>  
  9.         </property>  
  10.     </bean>  
  11.     <bean name="mydomain:myobj=MyObjectMBean" class="org.shirdrn.spring.jmx.autodetect.MyObject">  
  12.         <property name="id">  
  13.             <value>90000000001</value>  
  14.         </property>  
  15.         <property name="name">  
  16.             <value>shirdrn</value>  
  17.         </property>  
  18.     </bean>  
  19. </beans>  

上面配置中,MBeanExporter會查找本地MBean Server,指定的探測模式autodetectModeName爲AUTODETECT_ALL,這也是MBeanExporter的默認值(這個屬性完全可以省略,不用配置),無需手動向MBean Server進行註冊,便能管理配置的MBean對象“mydomain:myobj=MyObjectMBean”。

對於探測模式autodetectModeName屬性,Spring提供了4個取值:

AUTODETECT_NONE           不啓用自動探測,需要手動向MBean Server進行註冊,即通過MBeanExporter的beans屬性進入註冊;

AUTODETECT_MBEAN         在當前IOC容器中進行查找MBean組件;

AUTODETECT_ASSEMBLER  設置根據MBeanInfoAssembler的策略進行探測;

AUTODETECT_ALL               自動探測,是AUTODETECT_MBEAN和AUTODETECT_ASSEMBLER的並集。

另外,Spring的MBeanExporter也提供了autodetect屬性,取值爲true和false,指定對MBean組件的探測行爲。

 

普通Spring Bean管理

對於一個普通的Spring Bean,也可以作爲MBean來進行管理,Spring可以很好地支持。如下面一個普通的Java類:

  1. package org.shirdrn.spring.jmx;  
  2.   
  3. import org.apache.log4j.Logger;  
  4.   
  5. public class MyJavaObject {  
  6.   
  7.     private static final Logger LOG = Logger.getLogger(MyJavaObject.class);  
  8.     private long id;  
  9.     private String name;  
  10.   
  11.     public MyJavaObject() {  
  12.         super();  
  13.     }  
  14.   
  15.     public MyJavaObject(long id, String name) {  
  16.         super();  
  17.         this.id = id;  
  18.         this.name = name;  
  19.     }  
  20.   
  21.     public long getId() {  
  22.         return id;  
  23.     }  
  24.   
  25.     public void setId(long id) {  
  26.         this.id = id;  
  27.     }  
  28.   
  29.     public String getName() {  
  30.         return name;  
  31.     }  
  32.   
  33.     public void setName(String name) {  
  34.         this.name = name;  
  35.     }  
  36.   
  37.     public String show() {  
  38.         StringBuffer sb = new StringBuffer().append("id=").append(id).append(  
  39.                 ", name=").append(name);  
  40.         LOG.info("show()=" + sb.toString());  
  41.         return sb.toString();  
  42.     }  
  43. }  

它並沒有實現MBean管理接口,可以通過MBeanExporter的beans屬性進行註冊,配置如下所示:

  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <beans  
  3.     xmlns="http://www.springframework.org/schema/beans"  
  4.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  5.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">  
  6.     <bean id="mbServer" class="org.springframework.jmx.export.MBeanExporter">  
  7.         <property name="beans">  
  8.             <map>  
  9.                 <entry key="mydomain:myjavaobj=MyJavaObject" value-ref="myobj"/>  
  10.             </map>  
  11.         </property>  
  12.     </bean>  
  13.     <bean name="myobj" class="org.shirdrn.spring.jmx.MyJavaObject">  
  14.         <property name="id">  
  15.             <value>90000000001</value>  
  16.         </property>  
  17.         <property name="name">  
  18.             <value>shirdrn</value>  
  19.         </property>  
  20.     </bean>  
  21. </beans>  

因爲org.shirdrn.spring.jmx.MyJavaObject沒有對應的MBean接口,所以默認情況下,該類中public的成員都會暴露出來,通過MBean Server可以管理。實際上,系統中MBean的某些屬性或方法可能不需要暴露給外部進行管理,爲了克服這種缺點,Spring提供了基於方法列表和接口定製的功能,可以將你所感興趣的屬性或方法暴露給外部管理。

 

基於方法列表和接口定製的MBean管理

對於上述普通的Java類MyJavaObject,可以通過定製接口和方法列表,來暴露MBean屬性或方法。例如,一個方法列表,可以在進行配置的時候指定,如下所示:

  1. <bean id="assembler" class="org.springframework.jmx.export.assembler.MethodNameBasedMBeanInfoAssembler">  
  2.     <property name="managedMethods"  value="setId,setName,show" />  
  3. </bean>  

上面通過org.springframework.jmx.export.assembler.MethodNameBasedMBeanInfoAssembler來進行方法列表的設置。而對於接口,可以定義一個接口,如下所示:

  1. package org.shirdrn.spring.jmx.selected.interfaces;  
  2.   
  3. public interface SelectedMethodsInterface {  
  4.     public long setId(long id);  
  5.     public void setName(String name);  
  6.     public void show();  
  7. }  

Spring配置,如下所示:

  1. <bean id="assembler" class="org.springframework.jmx.export.assembler.InterfaceBasedMBeanInfoAssembler">  
  2.     <property name="managedInterfaces">  
  3.         <list>  
  4.             <value>org.shirdrn.spring.jmx.selected.interfaces.SelectedMethodsInterface</value>  
  5.         </list>  
  6.     </property>  
  7. </bean>  

上面通過org.springframework.jmx.export.assembler.InterfaceBasedMBeanInfoAssembler,可以設置一個接口列表,來管理MBean暴露的接口。

對於上面兩種方式,都得到一個assembler實例,需要將其注入到org.springframework.jmx.export.MBeanExporter中,如下所示:

  1. <bean id="exporter" class="org.springframework.jmx.export.MBeanExporter">  
  2.     <property name="beans">  
  3.         <map>  
  4.             <entry key="mydomain:javaObj=MyJavaObject" value-ref="javaObject" />  
  5.         </map>  
  6.     </property>  
  7.     <property name="assembler" ref="assembler" />  
  8. </bean>  

 

基於註解的MBean管理

對於一個普通的Java類,作爲MBean需要被管理,可以通過註解指定要暴露的屬性和方法,示例如下:

  1. package org.shirdrn.spring.jmx.annotation;  
  2.   
  3. import org.apache.log4j.Logger;  
  4. import org.springframework.jmx.export.annotation.ManagedAttribute;  
  5. import org.springframework.jmx.export.annotation.ManagedOperation;  
  6. import org.springframework.jmx.export.annotation.ManagedResource;  
  7.   
  8. @ManagedResource(objectName = "annojmx:myjao=AnnotationObject", description = "MyJavaAnnotationObject")  
  9. public class MyJavaAnnotationObject {  
  10.       
  11.     private static final Logger LOG = Logger.getLogger(MyJavaAnnotationObject.class);  
  12.     private long id;  
  13.     private String name;  
  14.   
  15.     public MyJavaAnnotationObject() {  
  16.         super();  
  17.     }  
  18.   
  19.     public MyJavaAnnotationObject(long id, String name) {  
  20.         super();  
  21.         this.id = id;  
  22.         this.name = name;  
  23.     }  
  24.   
  25.     @ManagedAttribute  
  26.     public long getId() {  
  27.         return id;  
  28.     }  
  29.   
  30.     public void setId(long id) {  
  31.         this.id = id;  
  32.     }  
  33.   
  34.     public String getName() {  
  35.         return name;  
  36.     }  
  37.   
  38.     @ManagedAttribute  
  39.     public void setName(String name) {  
  40.         this.name = name;  
  41.     }  
  42.   
  43.     @ManagedOperation  
  44.     public String show() {  
  45.         StringBuffer sb = new StringBuffer().append("id=").append(id).append(  
  46.                 ", name=").append(name);  
  47.         LOG.info("show()=" + sb.toString());  
  48.         return sb.toString();  
  49.     }  
  50. }  

上面@ManagedResource表示指定該類的實例作爲MBean註冊到MBean Server中,然後可以通過對屬性和方法分別使用@ManagedAttribute和@ManagedOperation來指定暴露的屬性和方法。有關這些註解的詳細內容,可以查閱相關文檔。

下面是一個基本的配置內容:

  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"  
  4.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">  
  5.     <bean id="exporter" class="org.springframework.jmx.export.MBeanExporter">  
  6.         <property name="assembler" ref="assembler" />  
  7.         <property name="namingStrategy" ref="namingStrategy" />  
  8.     </bean>  
  9.     <bean id="assembler"  
  10.         class="org.springframework.jmx.export.assembler.MetadataMBeanInfoAssembler">  
  11.         <property name="attributeSource" ref="jmxas" />  
  12.     </bean>  
  13.     <bean id="namingStrategy"  
  14.         class="org.springframework.jmx.export.naming.MetadataNamingStrategy">  
  15.         <property name="attributeSource" ref="jmxas" />  
  16.     </bean>  
  17.     <bean id="jmxas" class="org.springframework.jmx.export.annotation.AnnotationJmxAttributeSource" />  
  18.     <bean id="javaObj"  
  19.         class="org.shirdrn.spring.jmx.annotation.MyJavaAnnotationObject">  
  20.         <property name="id" value="201122121200" />  
  21.         <property name="name" value="shirdrn" />  
  22.     </bean>  
  23. </beans>  

上面使用了Spring的MBeanExporter,可以看到,Spring配置內容相對較多。

Spring還提供兩種簡化配置的方式,

一種是,提供了org.springframework.jmx.export.annotation.AnnotationMBeanExporter,可以將上述配置大大簡化,等價的配置如下所示:

  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"  
  4.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">  
  5.     <bean class="org.springframework.jmx.export.annotation.AnnotationMBeanExporter"/>  
  6.     <bean name="myAnnoObject"  
  7.         class="org.shirdrn.spring.jmx.annotation.MyJavaAnnotationObject">  
  8.         <property name="id" value="201122121200" />  
  9.         <property name="name" value="shirdrn" />  
  10.     </bean>  
  11. </beans>  

另一種是,提供了<context:mbean-export />標籤,更加簡潔,等價的配置如下所示:

  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"  
  4.     xsi:schemaLocation="http://www.springframework.org/schema/beans   
  5.         http://www.springframework.org/schema/beans/spring-beans-2.5.xsd  
  6.         http://www.springframework.org/schema/context   
  7.         http://www.springframework.org/schema/context/spring-context-2.5.xsd">  
  8.     <context:mbean-export registration="failOnExisting" />  
  9.     <bean name="myAnnoObject"  
  10.         class="org.shirdrn.spring.jmx.annotation.MyJavaAnnotationObject">  
  11.         <property name="id" value="201122121200" />  
  12.         <property name="name" value="shirdrn" />  
  13.     </bean>  
  14. </beans>  

 

通過遠程代理訪問MBean

根據JavaSE 6平臺的JMX技術架構定義,分爲設備層、代理層、遠程管理層這三層,我們通過可以使用Spring提供的org.springframework.jmx.access.MBeanProxyFactoryBean實現,定義代理,來訪問MBean Server管理MBean資源。實際上,Spring提供的這個代理功能位於JMX架構的遠程管理層,那麼在代理層和遠程管理層之間,要定義連接器,才能通過遠程管理層訪問到代理層的MBean Server組件。下面通過實例來實現:

首先,定義個普通的Java類,作爲待管理的MBean,如下所示:

  1. package org.shirdrn.spring.jmx.rmi;  
  2.   
  3. import org.apache.log4j.Logger;  
  4.   
  5. public class MyJavaObject {  
  6.       
  7.     private static final Logger LOG = Logger.getLogger(MyJavaObject.class);  
  8.     private long id;  
  9.     private String name;  
  10.   
  11.     public MyJavaObject() {  
  12.         super();  
  13.     }  
  14.   
  15.     public MyJavaObject(long id, String name) {  
  16.         super();  
  17.         this.id = id;  
  18.         this.name = name;  
  19.     }  
  20.   
  21.     public long getId() {  
  22.         return id;  
  23.     }  
  24.   
  25.     public void setId(long id) {  
  26.         this.id = id;  
  27.     }  
  28.   
  29.     public String getName() {  
  30.         return name;  
  31.     }  
  32.   
  33.     public void setName(String name) {  
  34.         this.name = name;  
  35.     }  
  36.   
  37.     public String show() {  
  38.         StringBuffer sb = new StringBuffer().append("id=").append(id).append(  
  39.                 ", name=").append(name);  
  40.         LOG.info("show()=" + sb.toString());  
  41.         return sb.toString();  
  42.     }  
  43. }  

其次,定義了一個Java接口,來按需暴露操作MBean資源的方法:

  1. package org.shirdrn.spring.jmx.rmi;  
  2.   
  3. public interface MyManagedInterfaces {  
  4.     public long setId(long id);  
  5.     public String getName();  
  6.     public String show();  
  7. }  

再次,看一下我們模擬代理層Spring配置server.xml內容:

  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  4.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">  
  5.     <bean id="exporter" class="org.springframework.jmx.export.MBeanExporter">  
  6.         <property name="beans">  
  7.             <map>  
  8.                 <entry key="mydomain:myjavaobj=MyJavaObject" value-ref="myJavaObj" />  
  9.             </map>  
  10.         </property>  
  11.         <property name="assembler" ref="assembler" />  
  12.     </bean>  
  13.     <bean id="assembler"  
  14.         class="org.springframework.jmx.export.assembler.InterfaceBasedMBeanInfoAssembler">  
  15.         <property name="managedInterfaces">  
  16.             <list>  
  17.                 <value>org.shirdrn.spring.jmx.rmi.MyManagedInterfaces</value>  
  18.             </list>  
  19.         </property>  
  20.     </bean>     
  21.     <bean name="myJavaObj" class="org.shirdrn.spring.jmx.rmi.MyJavaObject">  
  22.         <property name="id" value="88000000001" />  
  23.         <property name="name" value="shirdrn" />  
  24.     </bean>  
  25.     <bean id="server" class="org.springframework.jmx.support.ConnectorServerFactoryBean" depends-on="registry">  
  26.         <property name="objectName">  
  27.             <value>connector:name=rmi</value>  
  28.         </property>  
  29.         <property name="serviceUrl">  
  30.             <value>service:jmx:rmi://localhost/jndi/rmi://localhost:1099/jmxrmi</value>  
  31.         </property>  
  32.     </bean>  
  33.     <bean id="registry" class="org.springframework.remoting.rmi.RmiRegistryFactoryBean">  
  34.         <property name="port">  
  35.             <value>1099</value>  
  36.         </property>  
  37.     </bean>  
  38. </beans>  

上半部分配置已經非常熟悉,就是將MBean註冊到MBean Server中。下半部分是有關連接器的配置,它是基於RMI協議來進行適配,通過Spring提供的org.springframework.jmx.support.ConnectorServerFactoryBean來實現器,這樣,遠程管理層可以連接到啓動的連接服務器,來訪問MBean資源。

然後,我們啓動MBean Server及其連接服務器,代碼如下所示:

  1. package org.shirdrn.spring.jmx.rmi;  
  2.   
  3. import org.apache.log4j.Logger;  
  4. import org.springframework.context.support.ClassPathXmlApplicationContext;  
  5.   
  6. public class Server {  
  7.       
  8.     private static final Logger LOG = Logger.getLogger(Server.class);  
  9.       
  10.     public static void main(String[] args) throws Exception {  
  11.         new ClassPathXmlApplicationContext("org/shirdrn/spring/jmx/rmi/server.xml");  
  12.         LOG.info("Server started.");  
  13.           
  14.         Object lock = new Object();  
  15.         synchronized (lock) {  
  16.             lock.wait();  
  17.         }  
  18.     }  
  19. }  

啓動後,我們定義的MBean已經註冊到本地MBean Server中,同時啓動了連接器,監聽1099端口。

接着,我們定義遠程訪問層的Spring配置proxy.xml內容,如下所示:

  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  4.     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">  
  5.     <bean id="proxy" class="org.springframework.jmx.access.MBeanProxyFactoryBean">  
  6.         <property name="connectOnStartup" value="true" />  
  7.         <property name="objectName" value="mydomain:myjavaobj=MyJavaObject" />  
  8.         <property name="proxyInterface">  
  9.             <value>org.shirdrn.spring.jmx.rmi.MyManagedInterfaces</value>  
  10.         </property>  
  11.         <property name="serviceUrl">  
  12.             <value>service:jmx:rmi://localhost/jndi/rmi://localhost:1099/jmxrmi</value>  
  13.         </property>  
  14.     </bean>  
  15. </beans>  

最後,啓動代理,訪問MBean資源,代碼如下所示:

  1. package org.shirdrn.spring.jmx.rmi;  
  2.   
  3. import org.apache.log4j.Logger;  
  4. import org.springframework.context.ApplicationContext;  
  5. import org.springframework.context.support.ClassPathXmlApplicationContext;  
  6.   
  7. public class Proxy {  
  8.       
  9.     private static final Logger LOG = Logger.getLogger(Proxy.class);  
  10.       
  11.     public static void main(String[] args) throws Exception {  
  12.         ApplicationContext ctx = new ClassPathXmlApplicationContext(  
  13.                 "org/shirdrn/spring/jmx/rmi/proxy.xml");  
  14.         LOG.info("Proxy server started.");  
  15.         MyManagedInterfaces proxy = (MyManagedInterfaces) ctx.getBean("proxy");  
  16.         String message = proxy.show();  
  17.         LOG.info("proxy.show() = " + message);  
  18.           
  19.         Object lock = new Object();  
  20.         synchronized (lock) {  
  21.             lock.wait();  
  22.         }  
  23.     }  
  24. }  

這裏,只能訪問通過org.shirdrn.spring.jmx.rmi.MyManagedInterfaces接口定義的方法來操作註冊到MBean Server中的MBean資源。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章