本文转载自:http://bhsc-happy.iteye.com/blog/678163 尊重原创
JMX
Java Management Extensions,Java管理扩展,初步看了些资料,以为是专门管理,监控jvm的一些信息的,特别是visual VM这个监控jvm的东西,还有一个添加JMX连接的时候(我自己想错了,那样的话应该叫jvm Management Extensions),其实他能使得基于java语言开发的程序能被管理,并且是可扩展的。
Jdk以前是通过JVMPI之类来监测Java程序运行中的jvm和系统的一系列情况,现在通过jmx就可以做到,这是通过java.lang.management 包来实现的,这个包是 JMX 在 JDK方面 的一个应用,并不是表示jmx就是一个监控jvm的东西。
我们可以用jmx来监控我们的系统,通过公布API的方式,但是,这里采用监控这个词,也是受了前面的例子的影响,实际上,个人觉得,就可以用jmx来开发我们的系统。
现在的jboss,hibernate,tomcat各种应用都号称实现了JMX规范,将可管理,可调用的MBean注册到MBeanServer中,通过一种类似“web服务”的方式公布出去,并且伴有一个名字,可以通过该名字找到该MBean。并且,这里的MBean是可以被管理的,说到这里又想到了OSGI。
JMX与Web Service
个人认为,我们实现JMX规范,将东西发布出去,和通过web Service的方式是很类似的,也是可以远程调用的,只是相对的web Service的方式更加SOA一些,不过JMX号称也要提供对非java客户端的支持,也就是跨语言了吧。。。
现在的JMX连接方式:
JMXServiceURL url = new JMXServiceURL(“service:jmx:rmi:///jndi/rmi://localhost:9999/server”);
JMXConnector jmxc = JMXConnectorFactory.connect(url, null);
看了下源码,貌似还是通过RMI来实现的,不知道它要怎么实现非java客户端支持。
从这里,我觉得JMX可以实现的,我们也都可以通过web Service实现,只是看在它有个“M”上,以后如果有什么系统管理,监控方面的,可以考虑使用它,也许开发,个人觉得还是使用web service好一些。
说到这里,感觉OSGI与JMX也好像,在看到JMX能够对MBean进行管理的时候,我就觉得跟OSGI很像,OSGI管理的是Bundle,找了找资源,原来早就有人考虑过了:
http://teamojiao.iteye.com/blog/438334
顺便,在查资料的时候,发现一个东西,
if your question means, how to manage an OSGi runtime with JMX, you should have a look at MAEXO (http://code.google.com/p/maexo/). With MAEXO bundles up and running you will transparently get MBeans for a fair amount of services of the OSGi runtime as well as MBeans for your own services and bundles. Just have a look at the screencast.
摘一些话:仅做参考
<网友回复>
一个大系统中,各内部模块系统之间的基于接口方式的互相调用和治理,使用JMX是最佳方案.
带来的好处是
1.面向接口,远程调用对于开发人员是透明的,模块在调用JMX接口时,与调用本地方法几乎相同.
2.可视化的治理界面, 通过Jconsole等JMX客户端,可以实时监控系统,并且可实时调用方法进行某些操作.
典型应用场景:
某聊天系统,一台服务器作为 在线用户列表服务器 A1,N台服务器为用户提供聊天业务处理 N1 ,N2,N3…,
一台服务器作为后台治理系统A2.
系统治理员现在进行下面这样一个操作,察看某用户是否在线,找到该用户,发现其在线,则将该用户加入黑名单,并踢下线.
对应的JMX接口可以由以下几个:
A1为A2提供查询在线用户JMX接口,加入黑名单接口,kickout接口,
A1为N1..等服务器提供以下接口:注册业务服务器,添加在线用户.查找黑名单用户
N1…到N3为A1提供kickout接口.
因此在上面的踢下线操作,则由用户在A2的WEB界面发出,交由A1执行,A1记录黑名单之后,再找到用户所在业务服务器调用N1提供的接口让用户下线.
以上情形是在生产环境下的部署,而在开发工作,则可以将A1,A2,N…N3等功能合并在一个应用中调试.由于使用的是JMX接口,在本地调试合并之后,可以直接调用应用内部接口方法.
这样借助JMX实现的应用模块的灵活组装与拆分,使得系统的可以根据负载需要,根据性能情况,灵活的拆分和整合部署分布式的应用.
替代方案,选择webservice,xmlrpc等,但是这些都需要手工编写或用工具生成大量的代码来辅助完成接口间的java对象序列化 。
经典JMX案例:
1.jboss.使用JMX治理内部的各个Service。
2.基于java的开源网管软件 Hyperic HQ ,通过JMX与各被治理资源进行通讯和信息采集.
<网友回复>
JMX是一个治理的框架。
当我们想使用JMX的时候,就要问,我们的系统当中有需要监控治理的资源或者对象吗?实事求是一点,我们不能为了想使用一个高端的技术,就歪曲系统的本来面目。
假如第一个问题是肯定的,接下来就是看这些资源是否有生命周期。
经典案例:jboss就是将所有可部署的组件作为资源来治理,这些组建都有其生命周期。这个理念甚至延伸到了其系统内部,将其内部的服务作为组件纳入到 JMX中来,成就了jboss基于jmx的微内核系统。
软件包 java.lang.management
提供管理接口,用于监视和管理 Java 虚拟机以及 Java 虚拟机在其上运行的操作系统。
接口摘要
- ClassLoadingMXBean
用于 Java 虚拟机的类加载系统的管理接口。
- CompilationMXBean
用于 Java 虚拟机的编译系统的管理接口。
- GarbageCollectorMXBean
用于 Java 虚拟机的垃圾回收的管理接口。
- MemoryManagerMXBean
内存管理器的管理接口。
- MemoryMXBean
Java 虚拟机的内存系统的管理接口。
- MemoryPoolMXBean
内存池的管理接口。
- OperatingSystemMXBean
用于操作系统的管理接口,Java 虚拟机在此操作系统上运行。
- RuntimeMXBean
Java 虚拟机的运行时系统的管理接口。
- ThreadMXBean
Java 虚拟机线程系统的管理接口。 -
import java.lang.management.ClassLoadingMXBean; import java.lang.management.CompilationMXBean; import java.lang.management.GarbageCollectorMXBean; import java.lang.management.ManagementFactory; import java.lang.management.MemoryMXBean; import java.lang.management.MemoryManagerMXBean; import java.lang.management.MemoryPoolMXBean; import java.lang.management.MemoryUsage; import java.lang.management.OperatingSystemMXBean; import java.lang.management.RuntimeMXBean; import java.lang.management.ThreadMXBean; import java.util.List; import javax.management.MBeanServerConnection; public class MBeanDemo { public static void main(String[] args) { showJvmInfo(); showMemoryInfo(); showSystem(); showClassLoading(); showCompilation(); showThread(); showGarbageCollector(); showMemoryManager(); showMemoryPool(); } /** * Java 虚拟机的运行时系统 */ public static void showJvmInfo() { RuntimeMXBean mxbean = ManagementFactory.getRuntimeMXBean(); String vendor = mxbean.getVmVendor(); System.out.println("jvm name:" + mxbean.getVmName()); System.out.println("jvm version:" + mxbean.getVmVersion()); System.out.println("jvm bootClassPath:" + mxbean.getBootClassPath()); System.out.println("jvm start time:" + mxbean.getStartTime()); } /** * Java 虚拟机的内存系统 */ public static void showMemoryInfo() { MemoryMXBean mem = ManagementFactory.getMemoryMXBean(); MemoryUsage heap = mem.getHeapMemoryUsage(); System.out.println("Heap committed:" + heap.getCommitted() + " init:" + heap.getInit() + " max:" + heap.getMax() + " used:" + heap.getUsed()); } /** * Java 虚拟机在其上运行的操作系统 */ public static void showSystem() { OperatingSystemMXBean op = ManagementFactory.getOperatingSystemMXBean(); System.out.println("Architecture: " + op.getArch()); System.out.println("Processors: " + op.getAvailableProcessors()); System.out.println("System name: " + op.getName()); System.out.println("System version: " + op.getVersion()); System.out.println("Last minute load: " + op.getSystemLoadAverage()); } /** * Java 虚拟机的类加载系统 */ public static void showClassLoading(){ ClassLoadingMXBean cl = ManagementFactory.getClassLoadingMXBean(); System.out.println("TotalLoadedClassCount: " + cl.getTotalLoadedClassCount()); System.out.println("LoadedClassCount" + cl.getLoadedClassCount()); System.out.println("UnloadedClassCount:" + cl.getUnloadedClassCount()); } /** * Java 虚拟机的编译系统 */ public static void showCompilation(){ CompilationMXBean com = ManagementFactory.getCompilationMXBean(); System.out.println("TotalCompilationTime:" + com.getTotalCompilationTime()); System.out.println("name:" + com.getName()); } /** * Java 虚拟机的线程系统 */ public static void showThread(){ ThreadMXBean thread = ManagementFactory.getThreadMXBean(); System.out.println("ThreadCount" + thread.getThreadCount()); System.out.println("AllThreadIds:" + thread.getAllThreadIds()); System.out.println("CurrentThreadUserTime" + thread.getCurrentThreadUserTime()); //......还有其他很多信息 } /** * Java 虚拟机中的垃圾回收器。 */ public static void showGarbageCollector(){ List<GarbageCollectorMXBean> gc = ManagementFactory.getGarbageCollectorMXBeans(); for(GarbageCollectorMXBean GarbageCollectorMXBean : gc){ System.out.println("name:" + GarbageCollectorMXBean.getName()); System.out.println("CollectionCount:" + GarbageCollectorMXBean.getCollectionCount()); System.out.println("CollectionTime" + GarbageCollectorMXBean.getCollectionTime()); } } /** * Java 虚拟机中的内存管理器 */ public static void showMemoryManager(){ List<MemoryManagerMXBean> mm = ManagementFactory.getMemoryManagerMXBeans(); for(MemoryManagerMXBean eachmm: mm){ System.out.println("name:" + eachmm.getName()); System.out.println("MemoryPoolNames:" + eachmm.getMemoryPoolNames().toString()); } } /** * Java 虚拟机中的内存池 */ public static void showMemoryPool(){ List<MemoryPoolMXBean> mps = ManagementFactory.getMemoryPoolMXBeans(); for(MemoryPoolMXBean mp : mps){ System.out.println("name:" + mp.getName()); System.out.println("CollectionUsage:" + mp.getCollectionUsage()); System.out.println("type:" + mp.getType()); } } /** * 访问 MXBean 的方法的三种方法 */ public static void visitMBean(){ //第一种直接调用同一 Java 虚拟机内的 MXBean 中的方法。 RuntimeMXBean mxbean = ManagementFactory.getRuntimeMXBean(); String vendor1 = mxbean.getVmVendor(); System.out.println("vendor1:" + vendor1); //第二种通过一个连接到正在运行的虚拟机的平台 MBeanServer 的 MBeanServerConnection。 MBeanServerConnection mbs = null; // Connect to a running JVM (or itself) and get MBeanServerConnection // that has the JVM MXBeans registered in it /* try { // Assuming the RuntimeMXBean has been registered in mbs ObjectName oname = new ObjectName(ManagementFactory.RUNTIME_MXBEAN_NAME); String vendor2 = (String) mbs.getAttribute(oname, "VmVendor"); System.out.println("vendor2:" + vendor2); } catch (Exception e) { e.printStackTrace(); } */ //第三种使用 MXBean 代理 // MBeanServerConnection mbs3 = null; // RuntimeMXBean proxy; // try { // proxy = ManagementFactory.newPlatformMXBeanProxy(mbs3,ManagementFactory.RUNTIME_MXBEAN_NAME, // RuntimeMXBean.class); // String vendor = proxy.getVmVendor(); // } catch (IOException e) { // e.printStackTrace(); // } } }
输出:
jvm name:Java HotSpot(TM) Client VM jvm version:1.6.0-b105 jvm bootClassPath:C:\Program Files\Java\jdk1.6.0\jre\lib\resources.jar;C:\Program Files\Java\jdk1.6.0\jre\lib\rt.jar;C:\Program Files\Java\jdk1.6.0\jre\lib\sunrsasign.jar;C:\Program Files\Java\jdk1.6.0\jre\lib\jsse.jar;C:\Program Files\Java\jdk1.6.0\jre\lib\jce.jar;C:\Program Files\Java\jdk1.6.0\jre\lib\charsets.jar;C:\Program Files\Java\jdk1.6.0\jre\classes jvm start time:1307440032774 Heap committed:5177344 init:0 max:66650112 used:632640 Architecture: x86 Processors: 2 System name: Windows XP System version: 5.1 Last minute load: -1.0 TotalLoadedClassCount: 381 LoadedClassCount381 UnloadedClassCount:0 TotalCompilationTime:3 name:HotSpot Client Compiler ThreadCount5 AllThreadIds:[J@47b480 CurrentThreadUserTime15625000 name:Copy CollectionCount:0 CollectionTime0 name:MarkSweepCompact CollectionCount:0 CollectionTime0 name:CodeCacheManager MemoryPoolNames:[Ljava.lang.String;@1389e4 name:Copy MemoryPoolNames:[Ljava.lang.String;@c20e24 name:MarkSweepCompact MemoryPoolNames:[Ljava.lang.String;@2e7263 name:Code Cache CollectionUsage:null type:Non-heap memory name:Eden Space CollectionUsage:init = 917504(896K) used = 0(0K) committed = 0(0K) max = 4194304(4096K) type:Heap memory name:Survivor Space CollectionUsage:init = 65536(64K) used = 0(0K) committed = 0(0K) max = 458752(448K) type:Heap memory name:Tenured Gen CollectionUsage:init = 4194304(4096K) used = 0(0K) committed = 0(0K) max = 61997056(60544K) type:Heap memory name:Perm Gen CollectionUsage:init = 12582912(12288K) used = 0(0K) committed = 0(0K) max = 67108864(65536K) type:Non-heap memory name:Perm Gen [shared-ro] CollectionUsage:init = 8388608(8192K) used = 0(0K) committed = 0(0K) max = 8388608(8192K) type:Non-heap memory name:Perm Gen [shared-rw] CollectionUsage:init = 12582912(12288K) used = 0(0K) committed = 0(0K) max = 12582912(12288K) type:Non-heap memory