Tomcat从5.0版本开始引入,力图使JMX成为Tomcat未来版本的管理工具和平台。首先,让我们来对JMX做一个简单了解。JMX是JavaManagementExtension的缩写,可译为Java管理工具扩展,扩展的意思就是JMX不包含在标准的J2SE中,我们必须要另外下载JMXRI的实现。不过,这种把JMX排除在J2SE之外的情况已经成为历史了,J2SE5.0和J2SE6.0都已经包含了JMX的标准实现。这说明,JMX已经成为J2SE不可分割的一部分,另外一方面,JMX已经成为Java平台上管理工具的事实标准,在业界广泛使用。例如,JBOSS就是以JMX为微内核,Web应用模块和其它功能模块都可热插拨到这个微内核,将JMX的管理功能发挥得淋漓尽致。从当前业界使用情况看,JMX中的X(Extension,扩展)应该去掉,改名为JavaManagementStandardPlatform(JMSP,Java管理标准平台)更加合适。为了向下兼容,我们姑且还是称之为JMX吧。
JMX要管理的对象是什么呢,是资源。什么是资源,资源是指企业中的的各种应用软件和平台,举例来说,一个公司内部可能有许多应用服务器、若干Web服务器、一台至多台的数据库服务器及文件服务器等等,那么,如果我们想监视数据库服务器的内存使用情况,或者我们想更改应用服务器上JDBC最大连接池的数目,但我们又不想重启数据库和应用服务器,这就是典型意义上的资源管理,即对我们的资源进行监视(Monitoring,查看)和管理(Management,更改),这种监视和更改不妨碍当前资源的正常运行。对资源进行适当的监测和管理,可以让我们的IT资源尽可能的平稳运行,可以为我们的客户提供真正意思上的24×7服务。在资源耗尽或者在硬件出故障之前,我们就可以通过管理工具监测到,并通过管理工具进行热调整和插拔。独孤九剑,料敌机先,适当的资源管理就是我们料敌机先的工具,可以让我们立于IT服务的不败之地。在Sun公司提出JMX(JSR174)以前,人们通常都是使用SNMP对网络上的资源进行管理。SNMP的主要问题是入门门槛太高,不容易使用。所以Sun提出了JSR174倡议并且提供了一套JMX的参考实现。
从技术上说,JMX整体架构可分为三层,即资源植入层(InstrumentationLevel,可能有更好的译法?)、代理层(AgentLevel)和管理层(ManagerLevel),简述如下:
资源植入层(InstrumentationLevel):该层包含MBeans及这些MBeans所管理的资源,MBeans是一个Java对象,它必须实现JMX规范中规定的接口。按照JMX规范,在MBeans对象的接口中,我们可以指定管理层可以访问资源的哪些属性,可以调用资源的哪些方法,并且,在资源的属性发生变化是,我们的MBeans可以发出消息,通知对这些属性变化感兴趣的其它对象。JMX规范定义了四种MBeans,它们分别是标准MBeans(StandardMBeans)、动态MBeans(DynamicMBeans)、开放MBeans(OpenMBeans)和模态MBeans(ModelMBeans)。
代理层(AgentLevel):代理层的目的就是要把MBeans中实现的接口暴露给管理层,该层通常由MBeanServer和AgentServices构成,MBeanServer就是一个MBeans对象注册器,所有的资源MBeans都注册到这个MBeanServer,对象管理器或者其它的管理层应用程序可以通过访问MBeanServer,从而可以访问MBeanServer中注册的MBeans,当然也就可以监视和管理和这些MBeans绑定的资源。
管理层(ManagerLevel):又称之为分布式服务层(DistributedServices),顾名思义,该层主要包含一些管理应用程序,这些程序可以访问和操作JMX代理层(AgentLevel)。这些管理应用程序可以是一个Web应用,也可能是一个JavaSWT应用程序。
下面,我们举一个简单的例子,理解一下JMX中中的各个概念。我们家有一个中央热水系统(CentralHeaterSystem),它是我们家的一个资源,现在我们想通过JMI进行管理。现有的代码如下所示,当然,为简单起见,我们略去了一些JNI调用代码,因为厂家提供的API是用C语言写的。
a)热水器接口(CentralHeaterInf.java)的现有代码:
public interface CentralHeaterInf { public final static String HEATER_PROVIDER = "British Gas Company" ; public int getCurrentTemperature(); public void setCurrentTemperature( int newTemperature); public void turnOn(); public void turnOff(); }b) 热水器实现代码的现有代码 (CentralHeaterImpl .java )
public class CentralHeaterImpl implements CentralHeaterInf { private int currentTemperature ; public int getCurrentTemperature() { return currentTemperature ; } public void setCurrentTemperature( int newTemperature) { currentTemperature =newTemperature; } public void turnOff() { System. out .println( "The heater is off. " ); } public void turnOn() { System. out .println( "The heater is on. " ); } }
1.1 资源植入层 (Instrumentation Level) 代码示例
我们如何让JMX对我们的中央热水器进行管理呢?
首先,我们并不想让远程管理者能够关闭我们的中央热水器,因为热水器一旦关上,我们再也无法访问厂家提供的API。既然不能关闭它,我们的MBeans中也就不需要打开 (turnOn) 方法。所以,我们简单定义的 MBeans接口如下:
public interface CentralHeaterImplMBean { public String getHeaterProvider(); public int getCurrentTemperature(); public void setCurrentTemperature( int newTemperature); public String printCurrentTemperature(); }上面的MBean接口及其简单,意义也非常明显,我们只向管理程序公开热水器的生产厂家(该属性为只读,管理程序不能更改热水器的生产厂家),但管理程序可以获取并更改当前热水器的温度,并且可以打印出热水器的当前温度。
public class CentralHeaterImpl implements CentralHeaterInf,CentralHeaterImplMBean { int currentTemperature ; public int getCurrentTemperature() { return currentTemperature ; } public void setCurrentTemperature( int newTemperature) { currentTemperature =newTemperature; } public void turnOff() { System. out .println( "The heater is off. " ); } public void turnOn() { System. out .println( "The heater is on. " ); } public String getHeaterProvider() { return HEATER_PROVIDER ; } public String printCurrentTemperature() { String printMsg= "Current temperature is:" + currentTemperature ; System. out .println(printMsg); return printMsg; }}到此为止,我们的资源植入层 (Instrumentation Level) 的代码全部完成,它主要由一个 MBean(CentralHeaterImplBean) 及其实现类 CentralHeaterImpl 组成,在 CentralHeaterImplBean 这个 MBean 接口中,我们说明了要向管理程序暴露的属性和方法,在本例中,我们的管理程序可以访问热水器的生产厂家信息,同时还可以获取和设置并打印热水器的温度。在 MBean 的实现类中,我们实现了 MBean 接口中规定的所有方法。
import java.lang.management.ManagementFactory; import javax.management.MBeanServer; import javax.management.ObjectName; public class CentralHeaterAgent { private static MBeanServer mBeanServer ; public static void main(String[] args) throws Exception { ObjectName oname; // get the default MBeanServer from Management Factory mBeanServer = ManagementFactory.getPlatformMBeanServer (); // try { // create a instance of CentralHeaterImpl class CentralHeaterInf centralHeater = new CentralHeaterImpl(); // assign a Object name to above instance oname = new ObjectName( "MyHome:name=centralheater" ); // register the instance of CentralHeaterImpl to MBeanServer mBeanServer .registerMBean(centralHeater, oname); System. out .println( "Press any key to end our JMX agent..." ); System. in .read(); } }您可以看到,上面的代理层代码异常简单。前面讲过,代理层中最重要的对象就是MBeanServer,我们可以把MBeanServer理解为一个全局的HashMap,所有的MBeans都通过唯一的名字注册到这个HashMap,这个HashMap可以跨越JVM访问,甚至可以通过RMI、Http及其它手段跨越网络传输到其它机器,让其它机器也能访问这个MBeanServer中注册的对象。下面我们稍微理解一下代理层代码,在main()方法中,
-Dcom.sun.management.jmxremote -Dcom.sun.management.jmxremote.port=9999 -Dcom.sun.management.jmxremote.ssl="false" -Dcom.sun.management.jmxremote.authenticate="false"这四个 JVM 运行时参数的意义是, MBeanServer 允许其它管理程序通过 RMI 方式访问, RMI 端口是 9999, RMI 不使用 SSL 协议,也不需要验证。 Eclipse 的 Run 窗口如下:
-Dcom.sun.management.jmxremote -Dcom.sun.management.jmxremote.port=9999 -Dcom.sun.management.jmxremote.ssl="false" -Dcom.sun.management.jmxremote.authenticate="false"运行界面如下所示:
<role rolename="manager"/> <user username="admin" password="admin" roles="manager"/>这表示我们要添加一个新用户,用户名为admin,密码也是admin,用户具有manager权限。
ClassLoader classLoader = ClassLoaderFactory.createClassLoader (locations, types, parent); // Retrieving MBean server MBeanServer mBeanServer = null; if (MBeanServerFactory.findMBeanServer(null).size() > 0) { mBeanServer = (MBeanServer) MBeanServerFactory.findMBeanServer(null).get(0); } else { mBeanServer = MBeanServerFactory.createMBeanServer(); } // Register the server classloader ObjectName objectName = new ObjectName("Catalina:type=ServerClassLoader,name=" + name); mBeanServer.registerMBean(classLoader, objectName);4.1上面的代码首先使用ClassLoaderFactory工厂类创建一个ClassLoader;
public interface CentralHeater Decorator MBean { public String getHeaterProvider(); public int getCurrentTemperature(); public void setCurrentTemperature( int newTemperature); }CentralHeaterDecorator .java 的源代码:
public class CentralHeaterDecorator implements CentralHeaterDecoratorMBean { private CentralHeaterImpl centralHeater ; public CentralHeaterDecorator(CentralHeaterImpl theCentralHeater){ centralHeater =theCentralHeater; } public int getCurrentTemperature() { return centralHeater .getCurrentTemperature(); } public void setCurrentTemperature( int newTemperature) { centralHeater .setCurrentTemperature(newTemperature); } public String getHeaterProvider() { return centralHeater . HEATER_PROVIDER ; } public String printCurrentTemperature() { String returnMsg = "Current temperature is:" + centralHeater .getCurrentTemperature(); System. out .println(returnMsg); return returnMsg; } }最后,请把 Agent 代码中的下面一行:
CentralHeaterInf centralHeater = new CentralHeaterImpl();改为:
CentralHeaterDecoratorMBean centralHeater = new CentralHeaterDecorator(new CentralHeaterImpl());其运行效果完全一样,但我们完全没有改动既有代码。