JMX在Tomcat中的应用(四)

四、 Tomcat 中最简单的 MBean

下面我们打开 Tomcat 源代码,看看 Tomcat 中最简单的一个 MBean 。在 Tomcat 的启动引导类 Bootstrap.java 172 187 行,我们可以看到如下代码:

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

4.2 然后在 MBeanServerFactory 这个工厂类中查找 MBeanServer ,如果没有发现,就使用这个工厂类创建一个 MBeanServer

4.3 第三步是给刚才创建的 ClassLoader 这个 MBean 取个名字“ Name: Catalina:type=ServerClassLoader,name=common ”,然后注册到 MBeanServer

您如果在 JConsole 中观察这个 MBean ,会发现这个 MBean 没有向管理应用程序暴露任何属性和方法,并且 Classloader 似乎也符合 JMX 命名规范,它也不是一个 DynamicBean ,这是为什么呢?首先,这个 Classloader 其实一个 StandardClassloader ,而不是 JDK 中缺省的 Classloader ,您如果打开 ClassLoaderFactory createClassLoader 方法,马上就可以看到这一点;另外,看看 StanderClassloader 的类签名,我们会发现该类实现了 StandardClassLoaderMBean 接口,这是符合 JMX 命名规范的;请打开 StandardClassLoaderMBean 接口的源代码,您会发现这是一个空接口,这意味着实现这个接口的 MBean 不会向管理程序暴露任何属性和方法。

最后,希望这篇入门级的简单介绍,能有助于大家理解 JMX JMX Tomcat 中的应用。















附录:针对本文中的热水器小例子,我们给出了一个简单问题,即如何修改我们既有的代码,让这些代码所在的资源能使用
JMX 管理?如果您稍微翻阅一下 Tomcat 的源代码,您会发现, Tomcat 的作者们在 JMX 升级时对已有源代码的改动有点粗暴,勇猛有余,优美不足。 org.apache.catalina.core 包中的关键组件,大部分后加了 preRegister() getObjectName() 等等方法,在 init() 方法中又添加了一堆 Registry.getRegistry.unregisterComponent registerComponent 代码,这些方法其实这些 core 组件没有直接关系,也不是这些 core 组件应该具有的功能,并且这些后添加的代码及其类似。当然,这种情况在我们实际项目中更为多见,主要原因是时间不足,资源有限等等。

如果要比较优美的解决上面的问题,我个人认为,首先,保持现有代码,然后对现有代码进行扩展而不是大刀阔斧的修改已有代码。就拿我们热水器的简单例子来说,我们不要修改 CentralHeaterImpl.java 的现有代码,而是使用 Wrapper 设计模式,设计一个新类,然后将这个 CentralHeaterImpl 类适配成我们需要的 MBean 接口,具体实现如下:


a)   CentralHeaterDecoratorMBean.java 源代码

package
carl.test.jmx;
/**
* @author carlwu
*
*/
public interface CentralHeater Decorator MBean {

/**
* return the heater provider
* @return
*/
public String getHeaterProvider();


/**
* Get current temperature of heater
* @return the temperature of the heater
*/
public int getCurrentTemperature();

/**
* Set the new temperature
* @param newTemperature
*/
public void setCurrentTemperature( int newTemperature);

}

CentralHeaterDecorator .java 的源代码:

package
carl.test.jmx;
public class CentralHeaterDecorator implements CentralHeaterDecoratorMBean {

private CentralHeaterImpl centralHeater ;

public CentralHeaterDecorator(CentralHeaterImpl theCentralHeater){
centralHeater =theCentralHeater;
}


public int getCurrentTemperature() {
// TODO Auto-generated method stub
return centralHeater .getCurrentTemperature();
}

public void setCurrentTemperature( int newTemperature) {
// TODO Auto-generated method stub
centralHeater .setCurrentTemperature(newTemperature);
}



public String getHeaterProvider() {
// TODO Auto-generated method stub
return centralHeater . HEATER_PROVIDER ;
}


public String printCurrentTemperature() {
// TODO Auto-generated method stub
String returnMsg = "Current temperature is:"
    + centralHeater .getCurrentTemperature();
System. out .println(returnMsg);
return returnMsg;

}

}



最后,请把 Agent 代码中的下面一行:

CentralHeaterInf centralHeater =
new CentralHeaterImpl();

改为:

CentralHeaterDecoratorMBean centralHeater = new CentralHeaterDecorator(new CentralHeaterImpl());

其运行效果完全一样,但我们完全没有改动既有代码。

你可能感兴趣的:(apache,设计模式,jdk,tomcat,项目管理)