JXM簡(jiǎn)介
試想,一個(gè)正在運(yùn)行中的程序,我們?nèi)绻敫淖兂绦蛑械囊恍傩?,可以通過(guò)什么方法呢?可能有這么幾個(gè)方法:
- 對(duì)于服務(wù)器式的程序,可以制作管理頁(yè)面,通過(guò)HTTP post與servlet來(lái)更改服務(wù)器端程序的屬性。
- 對(duì)于服務(wù)器式的程序,還可以通過(guò)SOAP方式。但這需要程序開(kāi)啟了SOAP端的服務(wù)。
- 可以使用RMI遠(yuǎn)程調(diào)用。但這需要設(shè)計(jì)開(kāi)啟RMI服務(wù)。
- 如果是SWT或Swing的程序,則可以通過(guò)設(shè)計(jì)UI管理界面,使用戶(hù)可以和程序內(nèi)部交互。
- 還有一種方式,是將可改變的屬性放入配置文件XML,properties或數(shù)據(jù)庫(kù),程序輪詢(xún)配置文件,以求獲取最新的配置。
上面幾個(gè)方法都是常見(jiàn),但卻無(wú)法通用的。所謂通用,是指解決方案符合一個(gè)標(biāo)準(zhǔn),使得任何符合此標(biāo)準(zhǔn)的工具都能解析針對(duì)此標(biāo)準(zhǔn)的方案實(shí)現(xiàn)。這樣A公司設(shè)計(jì)的方案,B公司可以根據(jù)標(biāo)準(zhǔn)來(lái)解析。JMX就是Java管理標(biāo)準(zhǔn)。
對(duì)于一些參數(shù)的修改,網(wǎng)上有一段描述還是比較形象的:
- 程序初哥一般是寫(xiě)死在程序中,到要改變的時(shí)候就去修改代碼,然后重新編譯發(fā)布。
- 程序熟手則配置在文件中(JAVA一般都是properties文件),到要改變的時(shí)候只要修改配置文件,但還是必須重啟系統(tǒng),以便讀取配置文件里最新的值。
- 程序好手則會(huì)寫(xiě)一段代碼,把配置值緩存起來(lái),系統(tǒng)在獲取的時(shí)候,先看看配置文件有沒(méi)有改動(dòng),如有改動(dòng)則重新從配置里讀取,否則從緩存里讀取。
- 程序高手則懂得物為我所用,用JMX把需要配置的屬性集中在一個(gè)類(lèi)中,然后寫(xiě)一個(gè)MBean,再進(jìn)行相關(guān)配置。另外JMX還提供了一個(gè)工具頁(yè),以方便我們對(duì)參數(shù)值進(jìn)行修改。
JMX(Java Management Extensions)是一個(gè)為應(yīng)用程序植入管理功能的框架。JMX是一套標(biāo)準(zhǔn)的代理和服務(wù),實(shí)際上,用戶(hù)可以在任何Java應(yīng)用程序中使用這些代理和服務(wù)實(shí)現(xiàn)管理。JMX讓程序有被管理的功能,例如你開(kāi)發(fā)一個(gè)WEB網(wǎng)站,它是在24小時(shí)不間斷運(yùn)行,那么你肯定會(huì)對(duì)網(wǎng)站進(jìn)行監(jiān)控,如每天的UV、PV是多少;又或者在業(yè)務(wù)高峰的期間,你想對(duì)接口進(jìn)行限流,就必須去修改接口并發(fā)的配置值。
應(yīng)用場(chǎng)景:中間件軟件WebLogic的管理頁(yè)面就是基于JMX開(kāi)發(fā)的,而JBoss則整個(gè)系統(tǒng)都基于JMX構(gòu)架。
JMX的構(gòu)成
JMX由三部分組成:
基礎(chǔ)層:程序端的Instrumentation, 我把它翻譯成可操作的儀器。這部分就是指的MBean. MBean類(lèi)似于JavaBean。最常用的MBean則是Standard MBean和MXBean.
適配層:程序端的JMX agent. 這部分指的是MBean Server. MBean Server則是啟動(dòng)與JVM內(nèi)的基于各種協(xié)議的適配器。用于接收客戶(hù)端的調(diào)遣,然后調(diào)用相應(yīng)的MBeans.
-
接入層:客戶(hù)端的Remote Management. 這部分則是面向用戶(hù)的程序。此程序則是MBeans在用戶(hù)前投影,用戶(hù)操作這些投影,可以反映到程序端的MBean中去。這內(nèi)部的原理則是client通過(guò)某種協(xié)議調(diào)用agent操控MBeans.
JMX agent與Remote Management之間是通過(guò)協(xié)議鏈接的,這協(xié)議可能包含:- HTTP
- SNMP
- RMI
- IIOP
JMX agent中有針對(duì)上面協(xié)議的各種適配器??梢越馕鐾ㄟ^(guò)相應(yīng)協(xié)議傳輸過(guò)來(lái)的數(shù)據(jù)。Remote Management client則可以用現(xiàn)成的工具,如JConsole, 也可以自己書(shū)寫(xiě)java code。
實(shí)現(xiàn)一個(gè)JMX程序
1、 首先定義一個(gè)MBean接口,接口的命名規(guī)范為以具體的實(shí)現(xiàn)類(lèi)為前綴(這個(gè)規(guī)范很重要)
public interface HelloMBean {
public String getName();
public void setName(String name);
public String getAge();
public void setAge(String age);
public void helloWorld();
public void helloWorld(String str);
public void getTelephone();
}
2、定義一個(gè)實(shí)現(xiàn)類(lèi),實(shí)現(xiàn)上面的接口:
/*
* 該類(lèi)名稱(chēng)必須與實(shí)現(xiàn)的接口的前綴保持一致(即MBean前面的名稱(chēng)
*/
public class Hello implements HelloMBean {
private String name;
private String age;
public void getTelephone() {
System.out.println("get Telephone");
}
public void helloWorld() {
System.out.println("hello world");
}
public void helloWorld(String str) {
System.out.println("helloWorld:" + str);
}
public String getName() {
System.out.println("get name 123");
return name;
}
public void setName(String name) {
System.out.println("set name 123");
this.name = name;
}
public String getAge() {
System.out.println("get age 123");
return age;
}
public void setAge(String age) {
System.out.println("set age 123");
this.age = age;
}
}
3、定義agent層:
import java.lang.management.ManagementFactory;
import javax.management.JMException;
import javax.management.MBeanServer;
import javax.management.ObjectName;
public class HelloAgent {
public static void main(String[] args) throws JMException, Exception {
// 通過(guò)工廠(chǎng)類(lèi)獲取MBeanServer,用來(lái)做MBean的容器
MBeanServer server = ManagementFactory.getPlatformMBeanServer();
// ObjectName中的取名是有一定規(guī)范的,格式為:“域名:name=MBean名稱(chēng)”,其中域名和MBean的名稱(chēng)可以任意取。
ObjectName helloName = new ObjectName("jmxBean:name=hello");
//將Hello這個(gè)類(lèi)注入到MBeanServer中,注入需要?jiǎng)?chuàng)建一個(gè)ObjectName類(lèi)
server.registerMBean(new Hello(), helloName);
Thread.sleep(60*60*1000);
}
}
這樣,一個(gè)簡(jiǎn)單的JMX的DEMO已經(jīng)寫(xiě)完了,現(xiàn)在我們通過(guò)JDK提供的Jconsole來(lái)進(jìn)行操作。
4、在JDK安裝路徑 ·JAVA_HOME\bin· 下找到 jconsole.exe 這個(gè)小工具,雙擊打開(kāi)。
在本地進(jìn)程中找到 HelloAgent 并雙擊打開(kāi)
在當(dāng)前界面上,我們可以給程序中HelloMBean的屬性賦值,也可以調(diào)用其中的方法
這樣就做到動(dòng)態(tài)修改運(yùn)行中程序的狀態(tài)進(jìn)而管理程序。