J2EE總結(jié):Java命名與目錄接口JNDI
JNDI 是什么
JNDI是 Java 命名與目錄接口(Java Naming and Directory Interface),在J2EE規(guī)范中是重要的規(guī)范之一,不少專(zhuān)家認(rèn)為,沒(méi)有透徹理解JNDI的意義和作用,就沒(méi)有真正掌握J(rèn)2EE特別是EJB的知識(shí)。
那么,JNDI到底起什么作用?
要了解JNDI的作用,我們可以從“如果不用JNDI我們?cè)鯓幼??用了JNDI后我們又將怎樣做?”這個(gè)問(wèn)題來(lái)探討。
沒(méi)有JNDI的做法:
程序員開(kāi)發(fā)時(shí),知道要開(kāi)發(fā)訪問(wèn)MySQL數(shù)據(jù)庫(kù)的應(yīng)用,于是將一個(gè)對(duì) MySQL JDBC 驅(qū)動(dòng)程序類(lèi)的引用進(jìn)行了編碼,并通過(guò)使用適當(dāng)?shù)?JDBC URL 連接到數(shù)據(jù)庫(kù)。
就像以下代碼這樣:
- Connection conn=null;
- try {
- Class.forName("com.mysql.jdbc.Driver",
- true, Thread.currentThread().getContextClassLoader());
- conn=DriverManager.getConnection("jdbc:mysql://MyDBServer?user=qingfeng&password=mingyue");
- /* 使用conn并進(jìn)行SQL操作 */
- ......
- conn.close();
- }
- catch(Exception e) {
- e.printStackTrace();
- }
- finally {
- if(conn!=null) {
- try {
- conn.close();
- } catch(SQLException e) {}
- }
- }
這是傳統(tǒng)的做法,也是以前非Java程序員(如Delphi、VB等)常見(jiàn)的做法。這種做法一般在小規(guī)模的開(kāi)發(fā)過(guò)程中不會(huì)產(chǎn)生問(wèn)題,只要程序員熟悉Java語(yǔ)言、了解JDBC技術(shù)和MySQL,可以很快開(kāi)發(fā)出相應(yīng)的應(yīng)用程序。
沒(méi)有JNDI的做法存在的問(wèn)題:
1、數(shù)據(jù)庫(kù)服務(wù)器名稱(chēng)MyDBServer 、用戶(hù)名和口令都可能需要改變,由此引發(fā)JDBC URL需要修改;
2、數(shù)據(jù)庫(kù)可能改用別的產(chǎn)品,如改用DB2或者Oracle,引發(fā)JDBC驅(qū)動(dòng)程序包和類(lèi)名需要修改;
3、隨著實(shí)際使用終端的增加,原配置的連接池參數(shù)可能需要調(diào)整;
4、......
解決辦法:
程序員應(yīng)該不需要關(guān)心“具體的數(shù)據(jù)庫(kù)后臺(tái)是什么?JDBC驅(qū)動(dòng)程序是什么?JDBC URL格式是什么?訪問(wèn)數(shù)據(jù)庫(kù)的用戶(hù)名和口令是什么?”等等這些問(wèn)題,程序員編寫(xiě)的程序應(yīng)該沒(méi)有對(duì) JDBC 驅(qū)動(dòng)程序的引用,沒(méi)有服務(wù)器名稱(chēng),沒(méi)有用戶(hù)名稱(chēng)或口令 —— 甚至沒(méi)有數(shù)據(jù)庫(kù)池或連接管理。而是把這些問(wèn)題交給J2EE容器來(lái)配置和管理,程序員只需要對(duì)這些配置和管理進(jìn)行引用即可。
由此,就有了JNDI。
用了JNDI之后的做法:
首先,在在J2EE容器中配置JNDI參數(shù),定義一個(gè)數(shù)據(jù)源,也就是JDBC引用參數(shù),給這個(gè)數(shù)據(jù)源設(shè)置一個(gè)名稱(chēng);然后,在程序中,通過(guò)數(shù)據(jù)源名稱(chēng)引用數(shù)據(jù)源從而訪問(wèn)后臺(tái)數(shù)據(jù)庫(kù)。
具體操作如下(以JBoss為例):
1、配置數(shù)據(jù)源
在JBoss的 D:/jboss420GA/docs/examples/jca 文件夾下面,有很多不同數(shù)據(jù)庫(kù)引用的數(shù)據(jù)源定義模板。將其中的 mysql-ds.xml 文件Copy到你使用的服務(wù)器下,如 D:/jboss420GA/server/default/deploy。
修改 mysql-ds.xml 文件的內(nèi)容,使之能通過(guò)JDBC正確訪問(wèn)你的MySQL數(shù)據(jù)庫(kù),如下:
- <?xml version="1.0" encoding="UTF-8"?>
- <datasources>
- <local-tx-datasource>
- <jndi-name>MySqlDS</jndi-name>
- <connection-url>jdbc:mysql://localhost:3306/lw</connection-url>
- <driver-class>com.mysql.jdbc.Driver</driver-class>
- <user-name>root</user-name>
- <password>rootpassword</password>
- <exception-sorter-class-name>org.jboss.resource.adapter.jdbc.vendor.MySQLExceptionSorter</exception-sorter-class-name>
- <metadata>
- <type-mapping>mySQL</type-mapping>
- </metadata>
- </local-tx-datasource>
- </datasources>
這里,定義了一個(gè)名為MySqlDS的數(shù)據(jù)源,其參數(shù)包括JDBC的URL,驅(qū)動(dòng)類(lèi)名,用戶(hù)名及密碼等。
2、在程序中引用數(shù)據(jù)源:
- Connection conn=null;
- try {
- Context ctx=new InitialContext();
- Object datasourceRef=ctx.lookup("java:MySqlDS"); //引用數(shù)據(jù)源
- DataSource ds=(Datasource)datasourceRef;
- conn=ds.getConnection();
- /* 使用conn進(jìn)行數(shù)據(jù)庫(kù)SQL操作 */
- ......
- c.close();
- }
- catch(Exception e) {
- e.printStackTrace();
- }
- finally {
- if(conn!=null) {
- try {
- conn.close();
- } catch(SQLException e) { }
- }
- }
直接使用JDBC或者通過(guò)JNDI引用數(shù)據(jù)源的編程代碼量相差無(wú)幾,但是現(xiàn)在的程序可以不用關(guān)心具體JDBC參數(shù)了。
在系統(tǒng)部署后,如果數(shù)據(jù)庫(kù)的相關(guān)參數(shù)變更,只需要重新配置 mysql-ds.xml 修改其中的JDBC參數(shù),只要保證數(shù)據(jù)源的名稱(chēng)不變,那么程序源代碼就無(wú)需修改。
由此可見(jiàn),JNDI避免了程序與數(shù)據(jù)庫(kù)之間的緊耦合,使應(yīng)用更加易于配置、易于部署。
JNDI的擴(kuò)展:JNDI在滿(mǎn)足了數(shù)據(jù)源配置的要求的基礎(chǔ)上,還進(jìn)一步擴(kuò)充了作用:所有與系統(tǒng)外部的資源的引用,都可以通過(guò)JNDI定義和引用。
所以,在J2EE規(guī)范中,J2EE 中的資源并不局限于 JDBC 數(shù)據(jù)源。引用的類(lèi)型有很多,其中包括資源引用(已經(jīng)討論過(guò))、環(huán)境實(shí)體和 EJB 引用。特別是 EJB 引用,它暴露了 JNDI 在 J2EE 中的另外一項(xiàng)關(guān)鍵角色:查找其他應(yīng)用程序組件。
EJB 的 JNDI 引用非常類(lèi)似于 JDBC 資源的引用。在服務(wù)趨于轉(zhuǎn)換的環(huán)境中,這是一種很有效的方法??梢詫?duì)應(yīng)用程序架構(gòu)中所得到的所有組件進(jìn)行這類(lèi)配置管理,從 EJB 組件到 JMS 隊(duì)列和主題,再到簡(jiǎn)單配置字符串或其他對(duì)象,這可以降低隨時(shí)間的推移服務(wù)變更所產(chǎn)生的維護(hù)成本,同時(shí)還可以簡(jiǎn)化部署,減少集成工作。 外部資源”。
總 結(jié):
J2EE 規(guī)范要求所有 J2EE 容器都要提供 JNDI 規(guī)范的實(shí)現(xiàn)。JNDI 在 J2EE 中的角色就是“交換機(jī)” —— J2EE 組件在運(yùn)行時(shí)間接地查找其他組件、資源或服務(wù)的通用機(jī)制。在多數(shù)情況下,提供 JNDI 供應(yīng)者的容器可以充當(dāng)有限的數(shù)據(jù)存儲(chǔ),這樣管理員就可以設(shè)置應(yīng)用程序的執(zhí)行屬性,并讓其他應(yīng)用程序引用這些屬性(Java 管理擴(kuò)展(Java Management Extensions,JMX)也可以用作這個(gè)目的)。JNDI 在 J2EE 應(yīng)用程序中的主要角色就是提供間接層,這樣組件就可以發(fā)現(xiàn)所需要的資源,而不用了解這些間接性。
在 J2EE 中,JNDI 是把 J2EE 應(yīng)用程序合在一起的粘合劑,JNDI 提供的間接尋址允許跨企業(yè)交付可伸縮的、功能強(qiáng)大且很靈活的應(yīng)用程序。這是 J2EE 的承諾,而且經(jīng)過(guò)一些計(jì)劃和預(yù)先考慮,這個(gè)承諾是完全可以實(shí)現(xiàn)的。
最近一直在對(duì)J2EE的筆記進(jìn)行整理和復(fù)習(xí),雖然J2EE視頻是看過(guò)一遍了,但是當(dāng)我看自己做的筆記的時(shí)候陌生程度還是很大,而真正的對(duì)某個(gè)概念有所認(rèn)識(shí)的時(shí)候是將筆記和以前看過(guò)的視頻印象進(jìn)行摩擦,J2EE主要講解的內(nèi)容是各個(gè)規(guī)范,再清楚一些就是各個(gè)概念,現(xiàn)階段的目標(biāo)并不是掌握J(rèn)2EE,而是對(duì)J2EE進(jìn)行輪廓和概念上的了解和認(rèn)識(shí),到下一步DRP項(xiàng)目中再深層次的對(duì)各個(gè)規(guī)范進(jìn)行摩擦和認(rèn)識(shí)。
JNDI,翻譯為Java命名和目錄結(jié)構(gòu)(JavaNaming And Directory Interface)官方對(duì)其解釋為JNDI是一組在Java應(yīng)用中訪問(wèn)命名和目錄服務(wù)的API(ApplicationProgramming Interface)說(shuō)明很精煉,但是比較抽象。
上面的解釋中提高了命名服務(wù)和目錄服務(wù)兩個(gè)概念.先要了解JNDI就必須知道,命名服務(wù)和目錄服務(wù)是做什么用的。
學(xué)習(xí)新的概念和知識(shí),比較有效的方式是通過(guò)和以前所學(xué)過(guò)的內(nèi)容進(jìn)行聯(lián)系,比較。
關(guān)于命名服務(wù),其實(shí)我們很多時(shí)候都在用它,但是并不知道它是它,比較典型的是域名服務(wù)器DNS(Domain Naming Service),大對(duì)人對(duì)DNS還是比較了解的,它是將域名映射到IP地址的服務(wù).比如百度的域名www.baidu.com所映射的IP地址是http://202.108.22.5/,你在瀏覽器中輸入兩個(gè)內(nèi)容是到的同一個(gè)頁(yè)面.用命名服務(wù)器的原因是因?yàn)槲覀冇洃沚aidu這幾個(gè)有意義的字母要比記202.108.22.5更容易記憶,但如果站到計(jì)算機(jī)的角度上,它更喜歡處理這些數(shù)字。
從我們生活中找的話還有很多類(lèi)似的例子,比如說(shuō)你的身份證號(hào)和你的名字可以"理解"成一種命名服務(wù),你的學(xué)號(hào)和姓名也可以"解釋"為一種命名服務(wù)。
可以看出命名服務(wù)的特點(diǎn):一個(gè)值和另一個(gè)值的映射,將我們?nèi)祟?lèi)更容易認(rèn)識(shí)的值同計(jì)算機(jī)更容易認(rèn)識(shí)的值進(jìn)行一一映射。
到現(xiàn)在應(yīng)該對(duì)命名服務(wù)有所理解吧?
至于目錄服務(wù),從計(jì)算機(jī)角度理解為在互聯(lián)網(wǎng)上有著各種各樣的資源和主機(jī),但是這些內(nèi)容都是散落在互聯(lián)網(wǎng)中,為了訪問(wèn)這些散落的資源并獲得相應(yīng)的服務(wù),就需要用到目錄服務(wù)。
從我們?nèi)粘I钪腥ダ斫饽夸浄?wù)的概念可以從電話簿說(shuō)起,電話簿本身就是一個(gè)比較典型的目錄服務(wù),如果你要找到某個(gè)人的電話號(hào)碼,你需要從電話簿里找到這個(gè)人的名稱(chēng),然后再看其電話號(hào)碼。
理解了命名服務(wù)和目錄服務(wù)再回過(guò)頭來(lái)看JDNI,它是一個(gè)為Java應(yīng)用程序提供命名服務(wù)的應(yīng)用程序接口,為我們提供了查找和訪問(wèn)各種命名和目錄服務(wù)的通用統(tǒng)一的接口.通過(guò)JNDI統(tǒng)一接口我們可以來(lái)訪問(wèn)各種不同類(lèi)型的服務(wù).如下圖所示,我們可以通過(guò)JNDI API來(lái)訪問(wèn)剛才談到的DNS。
至此已經(jīng)對(duì)JNDI有了一個(gè)初步認(rèn)識(shí),如果想要進(jìn)一步了解JNDI,并對(duì)使用JDNI給我們帶來(lái)哪些便利之處,我推薦兩篇關(guān)于JDNI的文章,寫(xiě)的非常的好,兩篇文章從“如果不用JNDI我們?cè)鯓幼觯坑昧薐NDI后我們又將怎樣做?”這個(gè)角度來(lái)加深對(duì)JNDI的認(rèn)識(shí)。
原文鏈接:
http://blog.csdn.net/zhaosg198312/article/details/3979435
http://blog.csdn.net/jnqqls/article/details/7167517
【編輯推薦】