NDI 是什么
JNDI是 Java 命名與目錄接口(Java Naming and Directory Interface)恼策,在J2EE規(guī)范中是重要的規(guī)范之一腹鹉,不少專家認為牛哺,沒有透徹理解JNDI的意義和作用媳危,就沒有真正掌握J2EE特別是EJB的知識问潭。
那么萍启,JNDI到底起什么作用余寥?
要了解JNDI的作用怎爵,我們可以從“如果不用JNDI我們怎樣做永罚?用了JNDI后我們又將怎樣做啤呼?”這個問題來探討。
沒有JNDI的做法:
程序員開發(fā)時呢袱,知道要開發(fā)訪問MySQL數(shù)據(jù)庫的應(yīng)用官扣,于是將一個對 MySQL JDBC 驅(qū)動程序類的引用進行了編碼,并通過使用適當?shù)?JDBC URL 連接到數(shù)據(jù)庫羞福。就像以下代碼這樣:
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并進行SQL操作 */
......
conn.close();
}
catch(Exception e) {
e.printStackTrace();
}
finally {
if(conn!=null) {
try {
conn.close();
} catch(SQLException e) {}
}
}
這是傳統(tǒng)的做法惕蹄,也是以前非Java程序員(如Delphi、VB等)常見的做法治专。這種做法一般在小規(guī)模的開發(fā)過程中不會產(chǎn)生問題卖陵,只要程序員熟悉Java語言、了解JDBC技術(shù)和MySQL张峰,可以很快開發(fā)出相應(yīng)的應(yīng)用程序泪蔫。
沒有JNDI的做法存在的問題:
1、數(shù)據(jù)庫服務(wù)器名稱MyDBServer 喘批、用戶名和口令都可能需要改變撩荣,由此引發(fā)JDBC URL需要修改;
2饶深、數(shù)據(jù)庫可能改用別的產(chǎn)品餐曹,如改用DB2或者Oracle,引發(fā)JDBC驅(qū)動程序包和類名需要修改敌厘;
3凸主、隨著實際使用終端的增加,原配置的連接池參數(shù)可能需要調(diào)整额湘;
4卿吐、......
解決辦法:
程序員應(yīng)該不需要關(guān)心“具體的數(shù)據(jù)庫后臺是什么?JDBC驅(qū)動程序是什么锋华?JDBC URL格式是什么嗡官?訪問數(shù)據(jù)庫的用戶名和口令是什么?”等等這些問題毯焕,程序員編寫的程序應(yīng)該沒有對 JDBC 驅(qū)動程序的引用衍腥,沒有服務(wù)器名稱磺樱,沒有用戶名稱或口令 —— 甚至沒有數(shù)據(jù)庫池或連接管理。而是把這些問題交給J2EE容器來配置和管理婆咸,程序員只需要對這些配置和管理進行引用即可竹捉。
由此,就有了JNDI尚骄。
用了JNDI之后的做法:
首先块差,在在J2EE容器中配置JNDI參數(shù),定義一個數(shù)據(jù)源倔丈,也就是JDBC引用參數(shù)憨闰,給這個數(shù)據(jù)源設(shè)置一個名稱;然后需五,在程序中鹉动,通過數(shù)據(jù)源名稱引用數(shù)據(jù)源從而訪問后臺數(shù)據(jù)庫。
具體操作如下(以JBoss為例):
1宏邮、配置數(shù)據(jù)源
在JBoss的 D:/jboss420GA/docs/examples/jca 文件夾下面泽示,有很多不同數(shù)據(jù)庫引用的數(shù)據(jù)源定義模板。將其中的 mysql-ds.xml 文件Copy到你使用的服務(wù)器下蜜氨,如 D:/jboss420GA/server/default/deploy边琉。
修改 mysql-ds.xml 文件的內(nèi)容,使之能通過JDBC正確訪問你的MySQL數(shù)據(jù)庫记劝,如下:
<?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>
這里,定義了一個名為MySqlDS的數(shù)據(jù)源族扰,其參數(shù)包括JDBC的URL厌丑,驅(qū)動類名,用戶名及密碼等渔呵。
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進行數(shù)據(jù)庫SQL操作 */
......
c.close();
}
catch(Exception e) {
e.printStackTrace();
}
finally {
if(conn!=null) {
try {
conn.close();
} catch(SQLException e) { }
}
}
直接使用JDBC或者通過JNDI引用數(shù)據(jù)源的編程代碼量相差無幾,但是現(xiàn)在的程序可以不用關(guān)心具體JDBC參數(shù)了扩氢。在系統(tǒng)部署后耕驰,如果數(shù)據(jù)庫的相關(guān)參數(shù)變更,只需要重新配置 mysql-ds.xml 修改其中的JDBC參數(shù)录豺,只要保證數(shù)據(jù)源的名稱不變朦肘,那么程序源代碼就無需修改。
由此可見双饥,JNDI避免了程序與數(shù)據(jù)庫之間的緊耦合媒抠,使應(yīng)用更加易于配置、易于部署咏花。
JNDI的擴展:JNDI在滿足了數(shù)據(jù)源配置的要求的基礎(chǔ)上趴生,還進一步擴充了作用:所有與系統(tǒng)外部的資源的引用,都可以通過JNDI定義和引用。
所以苍匆,在J2EE規(guī)范中刘急,J2EE 中的資源并不局限于 JDBC 數(shù)據(jù)源。引用的類型有很多浸踩,其中包括資源引用(已經(jīng)討論過)叔汁、環(huán)境實體和 EJB 引用。特別是 EJB 引用民轴,它暴露了 JNDI 在 J2EE 中的另外一項關(guān)鍵角色:查找其他應(yīng)用程序組件攻柠。
EJB 的 JNDI 引用非常類似于 JDBC 資源的引用。在服務(wù)趨于轉(zhuǎn)換的環(huán)境中后裸,這是一種很有效的方法瑰钮。可以對應(yīng)用程序架構(gòu)中所得到的所有組件進行這類配置管理微驶,從 EJB 組件到 JMS 隊列和主題浪谴,再到簡單配置字符串或其他對象,這可以降低隨時間的推移服務(wù)變更所產(chǎn)生的維護成本因苹,同時還可以簡化部署苟耻,減少集成工作。 外部資源”扶檐。
總 結(jié):
J2EE 規(guī)范要求所有 J2EE 容器都要提供 JNDI 規(guī)范的實現(xiàn)凶杖。JNDI 在 J2EE 中的角色就是“交換機” —— J2EE 組件在運行時間接地查找其他組件、資源或服務(wù)的通用機制款筑。在多數(shù)情況下智蝠,提供 JNDI 供應(yīng)者的容器可以充當有限的數(shù)據(jù)存儲,這樣管理員就可以設(shè)置應(yīng)用程序的執(zhí)行屬性奈梳,并讓其他應(yīng)用程序引用這些屬性(Java 管理擴展(Java Management Extensions杈湾,JMX)也可以用作這個目的)。JNDI 在 J2EE 應(yīng)用程序中的主要角色就是提供間接層攘须,這樣組件就可以發(fā)現(xiàn)所需要的資源漆撞,而不用了解這些間接性。
在 J2EE 中于宙,JNDI 是把 J2EE 應(yīng)用程序合在一起的粘合劑浮驳,JNDI 提供的間接尋址允許跨企業(yè)交付可伸縮的、功能強大且很靈活的應(yīng)用程序捞魁。這是 J2EE 的承諾抹恳,而且經(jīng)過一些計劃和預(yù)先考慮,這個承諾是完全可以實現(xiàn)的署驻。
最近一直在對J2EE的筆記進行整理和復(fù)習,雖然J2EE視頻是看過一遍了,但是當我看自己做的筆記的時候陌生程度還是很大,而真正的對某個概念有所認識的時候是將筆記和以前看過的視頻印象進行摩擦,J2EE主要講解的內(nèi)容是各個規(guī)范,再清楚一些就是各個概念,現(xiàn)階段的目標并不是掌握J2EE,而是對J2EE進行輪廓和概念上的了解和認識,到下一步DRP項目中再深層次的對各個規(guī)范進行摩擦和認識奋献。
JNDI,翻譯為Java命名和目錄結(jié)構(gòu)(JavaNaming And Directory Interface)官方對其解釋為JNDI是一組在Java應(yīng)用中訪問命名和目錄服務(wù)的API(ApplicationProgramming Interface)說明很精煉,但是比較抽象健霹。上面的解釋中提高了命名服務(wù)和目錄服務(wù)兩個概念.先要了解JNDI就必須知道,命名服務(wù)和目錄服務(wù)是做什么用的。學習新的概念和知識瓶蚂,比較有效的方式是通過和以前所學過的內(nèi)容進行聯(lián)系糖埋,比較。
關(guān)于命名服務(wù),其實我們很多時候都在用它,但是并不知道它是它,比較典型的是域名服務(wù)器DNS(Domain Naming Service),大對人對DNS還是比較了解的,它是將域名映射到IP地址的服務(wù).比如百度的域名www.baidu.com所映射的IP地址是http://202.108.22.5/,你在瀏覽器中輸入兩個內(nèi)容是到的同一個頁面.用命名服務(wù)器的原因是因為我們記憶baidu這幾個有意義的字母要比記202.108.22.5更容易記憶,但如果站到計算機的角度上,它更喜歡處理這些數(shù)字窃这。
從我們生活中找的話還有很多類似的例子,比如說你的身份證號和你的名字可以"理解"成一種命名服務(wù),你的學號和姓名也可以"解釋"為一種命名服務(wù)瞳别。
可以看出命名服務(wù)的特點:一個值和另一個值的映射,將我們?nèi)祟惛菀渍J識的值同計算機更容易認識的值進行一一映射。
到現(xiàn)在應(yīng)該對命名服務(wù)有所理解吧?
至于目錄服務(wù),從計算機角度理解為在互聯(lián)網(wǎng)上有著各種各樣的資源和主機,但是這些內(nèi)容都是散落在互聯(lián)網(wǎng)中,為了訪問這些散落的資源并獲得相應(yīng)的服務(wù),就需要用到目錄服務(wù)杭攻。
從我們?nèi)粘I钪腥ダ斫饽夸浄?wù)的概念可以從電話簿說起,電話簿本身就是一個比較典型的目錄服務(wù),如果你要找到某個人的電話號碼,你需要從電話簿里找到這個人的名稱,然后再看其電話號碼祟敛。
理解了命名服務(wù)和目錄服務(wù)再回過頭來看JDNI,它是一個為Java應(yīng)用程序提供命名服務(wù)的應(yīng)用程序接口,為我們提供了查找和訪問各種命名和目錄服務(wù)的通用統(tǒng)一的接口.通過JNDI統(tǒng)一接口我們可以來訪問各種不同類型的服務(wù).如下圖所示,我們可以通過JNDI API來訪問剛才談到的DNS。
至此已經(jīng)對JNDI有了一個初步認識,如果想要進一步了解JNDI,并對使用JDNI給我們帶來哪些便利之處,我推薦兩篇關(guān)于JDNI的文章,寫的非常的好,兩篇文章從“如果不用JNDI我們怎樣做兆解?用了JNDI后我們又將怎樣做馆铁?”這個角度來加深對JNDI的認識。