1.什么是JMX
JMX可以监控类的运行情况,可以在程序运行期查看并修改类属性的信息。
举一个应用实例:在一个系统中常常会有一些配置信息,比如服务的IP地址,端口号什么的,那么如何来写这些代码呢?
下面是一段摘自http://blog.csdn.net/xiaojianpitt/article/details/2961695:
程序初哥一般是写死在程序里,到要改变时就去改程序,然后再编译发布;
程序熟手则一般把这些信息写在一个配置文件里(JAVA一般都是*.properties文件),到要改变时只要改配置文件,但还是重新启动系统,以便读取配置文件里的新值;
程序好手则会写一个段代码,把配置值缓存起来,系统在读值的时候,先看看配置文件有没有更动。如有更改则重读一遍,否则从缓存里读取值
程序高手则懂得取物为我所用,用JMX把配置属性集中在一个类,然后写一个叫MBean的东东,再配置一下就轻松搞定了。而且JMX自动提供了一个WEB页面来给你来改变这些配置信息。
使用JMX可以在不影响程序运行的情况下根据访问量去修改数据库连接池的大小。
我们可以自己编写代码基于JMX监控类的运行情况,也可以基于Spring去监控Bean的运行情况,还可以监控activemq,查看消息的接收情况。
2.自己编写JMX代码
编写接口,类名的结尾都是MBean
public interface HelloMBean { public String getName(); public void setName(String name); public void print(); public void print(String str); }
编写需要监控的类,继承HelloMBean接口
public class Hello implements HelloMBean { private String m_name = "scorpio"; @Override public String getName() { return m_name; } @Override public void setName(String name) { m_name = name; } @Override public void print() { System.out.println("Hello "+m_name); } @Override public void print(String str) { System.out.println("hello"+str); } }
编写监控类的代码,监控上面的java类
public class JmxService { public static void main(String[] args) { MBeanServer jmxServer = MBeanServerFactory.createMBeanServer(); MBeanServer jconsolJmxServer = ManagementFactory.getPlatformMBeanServer(); HelloMBean helloMBean = new Hello(); ObjectName helloName = null; try { //jmx:rmi:///jndi/rmi://localhost:8888/jmxrmi LocateRegistry.createRegistry(8888); helloName = new ObjectName("raisecom.com:name=scorpio,type=Echo"); jmxServer.registerMBean(helloMBean, helloName); jconsolJmxServer.registerMBean(helloMBean, helloName); JMXServiceURL jmxURL = new JMXServiceURL("service:jmx:rmi:///jndi/rmi://localhost:8888/server"); JMXConnectorServer jmxConnectorServer = JMXConnectorServerFactory.newJMXConnectorServer(jmxURL, null, jmxServer); jmxConnectorServer.start(); Thread.sleep(Long.MAX_VALUE); System.out.println("start"); } catch (MalformedObjectNameException e) { e.printStackTrace(); } catch (NullPointerException e) { e.printStackTrace(); } catch (InstanceAlreadyExistsException e) { e.printStackTrace(); } catch (MBeanRegistrationException e) { e.printStackTrace(); } catch (NotCompliantMBeanException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } catch (InterruptedException e) { e.printStackTrace(); } } }
可以修改name属性的值,再调用print方法就可以看到修改的结果。这样就可以在程序运行期动态地做出改变。
也可以基于RMI的方式登陆:
service:jmx:rmi:///jndi/rmi://localhost:8888/server
3.基于JMX监控Spring的Bean
<!-- 需要监控的类 --> <bean id = "mbeanExporter" class = "org.springframework.jmx.export.MBeanExporter"> <property name = "beans"> <map> <entry key = "spitter:name=Controller" value-ref = "baseController"/> </map> </property> <property name = "assembler" ref = "assembler"/> </bean> <!-- 控制需要监控类的属性 --> <bean id = "assembler" class = "org.springframework.jmx.export.assembler.MethodNameBasedMBeanInfoAssembler"> <property name = "managedMethods"> <set> <value>setPageSize</value> <value>getPageSize</value> </set> </property> </bean>
可以和自己写代码基于JMX监控和修改Java类的属性一样,我们可以通过JMX监控及修改Spring的Bean的属性。
4.使用JMX监控activemq
1、默认JMX功能是没有打开的,需要在activemq.xml的broker配置上添加useJmx="true"
2、需要在managementContext里,修改为createConnector="true",(同时这里也可以修改jmx的端口和domain)
默认情况下,ActiveMQ使用useJmx后,jmx的url为
service:jmx:rmi:///jndi/rmi://localhost:1099/jmxrmi
这时,jmx的MBean server被绑死在localhost上,无法在broker所在机器以外的机器访问。
其实ActiveMQ提供了jmx相关的几个配置,不仅可以调整这个url中的ip和端口,还可以使用其他的MBean server。
属性名称 | 默认值 | 描述 |
---|---|---|
useMBeanServer | true | 为true则避免创建一个MBeanServer,使用jvm中已有的MBeanServer |
jmxDomainName | org.apache.activemq | jmx域,所有ObjectName的前缀 |
createMBeanServer | true | 为true则在需要时创建一个MBeanServer |
createConnector | true | 为true则创建一个JMX connector |
connectorPort | 1099 | JMX connector的端口 |
connectorHost | localhost | JMX connector和RMI server(rmiServerPort>0)的host |
rmiServerPort | 0 | RMI server的端口(便于穿过防火墙) |
connectorPath | /jmxrmi | JMX connector注册的路径 |
findTigerMBeanServer | true | 启用或禁用查找Java 5 平台的 MBeanServer |
其他需要修改的地方如下:
ACTIVEMQ_SUNJMX_START="-Dcom.sun.management.jmxremote.port=8888" ACTIVEMQ_SUNJMX_START="$ACTIVEMQ_SUNJMX_START -Dcom.sun.management.jmxremote.password.file=${ACTIVEMQ_CONF}/jmx.password" ACTIVEMQ_SUNJMX_START="$ACTIVEMQ_SUNJMX_START -Dcom.sun.management.jmxremote.access.file=${ACTIVEMQ_CONF}/jmx.access" ACTIVEMQ_SUNJMX_START="$ACTIVEMQ_SUNJMX_START -Dcom.sun.management.jmxremote.ssl=false" ACTIVEMQ_SUNJMX_START="$ACTIVEMQ_SUNJMX_START -Dcom.sun.management.jmxremote" ACTIVEMQ_SUNJMX_CONTROL="service:jmx:rmi:///jndi/rmi://localhost:8888/server admin activemq"
修改完后使用JConsole既可以监控activemq消息的情况了。