一JMX技术
1.MBean的概念和名称
JMX中的MBean就是针对这些资源而设计的,MBean通过“属性”来显示资源的状态,通过“操作”来实现对资源的控制和管理。MBean并不一定直接去和资源的状态,只要通过这个MBean可以实现“属性”和“操作”就足够了。对于MBean的调用主要通过代理来实现,这里的代理是指软件系统中用于操纵MBean的组件。
每个MBean需要一个名称,用javax.management.ObjectName这个类来实现,以便将MBean之间区分开。MBean的名称包括两个部分:域名和键表。域名是MBean所在域的名称,如book.java 指的就是book下的 java子域。键表由若干个键名和键值组成,例如service=invoker,type=http,指的是名为service和type两个键,它们的健值是invoker和http。健表的惟一性与键之间的排列顺序无关。域名和键表之间通过“:”连接共同组成MBean的名称,如book.java:service=example。
2. MBean的类型
目前共有4种类型的MBean:标准MBean,动态MBean,开放MBean和模型MBean。MBean需要由代理来操纵。而代理是通过MBean服务器,即MBeanServer,来操纵每个MBean的。每个MBean都以来ObjectName注册到MBeanServer,代理再通过MBeanServer来调用MBean的属性和方法,调用的过程是借助Java的反射功能来实现的。每个MBean将有关自己的属性和方法的描述信息封装在MBeanInfo中,并向MBeanServer提供获取MBeanInfo的方法,即实现getMBeanInfo()方法或者通过一个确定的接口直接暴露自己的方法和属性。如果采取第一种方法来实现,那么这个MBean主是一种动态MBean;如果采用后一种方法来实现,那么这个MBean就是一种标准MBean。
2.1标准MBean
下面介绍一个标准MBean的例子:HelloMBean是接口,Hello 是实现类,在JMX中,标准MBean的接口和实现类之间通过名称后缀-MBean来相互对应。
package com.joy_cz.jmx;
/**
* @author joy_cz
*/
public interface HelloMBean {
public void sayHello();
public int getTimes();
public void setTimes(int t);
}
标准MBean就是用这种接口的方式来静态的提供MBean的方法和属性,下面是标准HelloMBean的实现类。
package com.joy_cz.jmx;
/**
* @author joy_cz
*/
public class Hello implements HelloMBean {
private int times;
/* (非 Javadoc)
* @see com.joy_cz.jmx.HelloMBean#getTimes()
*/
public int getTimes() {
// TODO 自动生成方法存根
return times;
}
/* (非 Javadoc)
* @see com.joy_cz.jmx.HelloMBean#sayHello()
*/
public void sayHello() {
// TODO 自动生成方法存根
System.out.println("say hello to you ~");
times++;
}
/* (非 Javadoc)
* @see com.joy_cz.jmx.HelloMBean#setTimes(int)
*/
public void setTimes(int t) {
// TODO 自动生成方法存根
times = t;
}
}
调用这个MBean需要一个代理程序。在代理程序HelloAgent中,Hello的对象实例以“book.java:service=example”的名称被注册到了MBeanServer,而后通过相同的ObjectName名称借助MBeanServer调用了Hello的“sayHello”方法,获取并设置了“Times”属性的值。
package com.joy_cz.jmx;
import javax.management.MBeanServer;
import javax.management.MBeanServerFactory;
import javax.management.ObjectName;
import javax.management.Attribute;;
/**
* @author Administrator
*
*/
public class HelloAgent {
/**
* @param args
*/
public static void main(String[] args) throws Exception{
// TODO 自动生成方法存根
MBeanServer server = MBeanServerFactory.newMBeanServer();
ObjectName name = new ObjectName("book.java:service=example");
Hello hello = new Hello();
server.registerMBean(hello, name);
Object[] opArgs1 = {};
String[] sig1={};
Object result1 = server.invoke(name, "sayHello", opArgs1,sig1);
Integer times1 = (Integer)server.getAttribute(name, "Times");
System.out.println(times1);
server.setAttribute(name, new Attribute("Times",new Integer(7)));
Integer times2 =(Integer)server.getAttribute(name, "Times");
System.out.println(times2);
}
}
看过HelloAgent的代码部分,你会发现代理程序在使用 sayHello和Times这样的固定名称来间接访问Hello的方法和属性。MBeanServer可以提供对应注册名称MBean的所有信息,通过getMBeanInfo方法,将MBean的信息封装在MBeanInfo类中,其中包含表示属性和方法的各种元数据类型。
MBeanInfo mbi = server.getMBeanInfo(name);
MBeanOperationInfo[ ] mbois = mbi.getOperations();
for(int i = 0;i<mbois.length;i++)
{
MBeanOperationInfo mboi = mbois[i];
System.out.println(mboi.getName());
}
MBeanAttributeInfo[ ] mbais = mbi.getAttributes();
for(int i = 0;i<mbais.length;i++)
{
MBeanAttributeInfo mbai = mbais[i];
System.out.println(mbai.getName());
}
2.2动态MBean
动态MBean的MBeanInfo是由实现类自己提供的,也就是说,动态MBean需要自己实现包括getMBeanInfo在内的下列方法:
MBeanInfo getMBeanInfo()
Object invoke(String actionName,Object[] params,String[] signature)
void setAttribute(Attribute attribute)
Object getAttribute(String attribute)
AttributeList setAttributes(AttributeList list)
AttributeList getAttributes(String [] attributes)
这些方法组成了DynamicMBean接口。所以动态MBean需要实现DynamicMBean接口。
例子:
/**
*
*/
package com.joy_cz.jmx;
import javax.management.Attribute;
import javax.management.AttributeList;
import javax.management.AttributeNotFoundException;
import javax.management.DynamicMBean;
import javax.management.InvalidAttributeValueException;
import javax.management.MBeanException;
import javax.management.MBeanInfo;
import javax.management.ReflectionException;
/**
* @author Administrator
*
*/
public class DynamicHello implements DynamicMBean {
/* (非 Javadoc)
* @see javax.management.DynamicMBean#getAttribute(java.lang.String)
*/
private int times;
/* (非 Javadoc)
* @see com.joy_cz.jmx.HelloMBean#getTimes()
*/
public int getTimes() {
// TODO 自动生成方法存根
return times;
}
/* (非 Javadoc)
* @see com.joy_cz.jmx.HelloMBean#sayHello()
*/
public void sayHello() {
// TODO 自动生成方法存根
System.out.println("say hello to you ~");
times++;
}
/* (非 Javadoc)
* @see com.joy_cz.jmx.HelloMBean#setTimes(int)
*/
public void setTimes(int t) {
// TODO 自动生成方法存根
times = t;
}
public Object getAttribute(String attribute)
throws AttributeNotFoundException, MBeanException,
ReflectionException {
// TODO 自动生成方法存根
return null;
}
/* (非 Javadoc)
* @see javax.management.DynamicMBean#getAttributes(java.lang.String[])
*/
public AttributeList getAttributes(String[] attributes) {
// TODO 自动生成方法存根
return null;
}
/* (非 Javadoc)
* @see javax.management.DynamicMBean#getMBeanInfo()
*/
public MBeanInfo getMBeanInfo() {
// TODO 自动生成方法存根
return null;
}
/* (非 Javadoc)
* @see javax.management.DynamicMBean#invoke(java.lang.String, java.lang.Object[], java.lang.String[])
*/
public Object invoke(String actionName, Object[] params, String[] signature)
throws MBeanException, ReflectionException {
// TODO 自动生成方法存根
return null;
}
/* (非 Javadoc)
* @see javax.management.DynamicMBean#setAttribute(javax.management.Attribute)
*/
public void setAttribute(Attribute attribute)
throws AttributeNotFoundException, InvalidAttributeValueException,
MBeanException, ReflectionException {
// TODO 自动生成方法存根
}
/* (非 Javadoc)
* @see javax.management.DynamicMBean#setAttributes(javax.management.AttributeList)
*/
public AttributeList setAttributes(AttributeList attributes) {
// TODO 自动生成方法存根
return null;
}
}
2.3开放MBean
开放MBean也是一生中 动态MBean。比一般的动态MBean强大的特点在于。开放MBean定义了一套数据类型,利用开放MBean的数据类型,MBean可以设定更加复杂的属性和操作方法参数。
在开放MBean的属性和参数中使用的数据类型都继承自OpenType抽象类。继承OpenType的类型有4种:ArrayType,CompositeType,SimpleType和TabularType。它们分别代表数组类型、复合类型、简单类型和表格类型。其中。简单类型包括Java的常用数据类型。开放MBean还提供了TabularData,CompositeData接口来表示复杂数据对象。
开放MBean需要实现DynamicMBean接口,并利用OpenMBeanInfo来提供MBean信息。
2.4模型MBean
模型MBean是对动态MBean的扩展。模型MBean需要实现的ModelMBean接口继承自DynamicMBean接口,ModelMBean还继承了ModelMBeanNotificationBroadcaster和PersisrentMBent这两个接口。ModelMBeanNotificationBroadcaster使模型MBent能够传递通知信息,模型MBent的属性改变可以直接触发JMX通知事件。PersistentMBean接口的实现使模型MBent的状态信息可以被持久化输出和导入。
3使用JMX服务
3.1通知服务
JMX还提供了MBent的通知机制,通知信息可以从实现了NotificationBroadcaster的MBent(例如模型MBent)发送至实现了NotificationListener的MBent,中间可以插入NotificationFilter,起到过滤信息的作用。
3.2监视服务
JMX主要提供了三种监视器:CounterMonitor,GaugeMonitor和StringMonitor。这三种监视器本身也是MBent,它们分别用于监视属性值的增长信息,监控属性值在一个区域内的情况,监控字符类型属性的匹配情况。当属性值在监控中匹配到改变信息后,监控器就会发出JMX通知给感兴趣的NotificationListener。
3.3动态加载MBean
运用JMX技术的另一个好处是让你可以很容易地将第三方的工作嵌入进来,特别是在JMX中,利用JMX提供的MBean动态加载服务,多个MBean的信息可以被记录到一个Mlet文件中,通过Mlet这个服务。可以从Mlet直接启动MBean。
<MLET
CODE = "org.mortbay.jetty.jmx.ServerMBean"
ARCHIVE = "org.mortbay.jetty.jmx.jar"
CODEBASE = "lib">
<ARG type = "java.lang.String" value = "etc/jetty.xml">
</MLET>
Mlet文件中的每个Mlet元素代表着一个MBean,其中记录着MBean的代码名称、所在的确jar文件位置、目录位置和若干个参数。有了这个Mlet文件,你只需使用下面的几行代码就可以启动jetty服务器。