------------
JNDI是 Java 命名與目錄接口(Java Naming and Directory Interface),在J2EE規(guī)范中是重要的規(guī)范之一,不少專家認(rèn)為,沒有透徹理解JNDI的意義和作用,就沒有真正掌握J(rèn)2EE特別是EJB的知識。

那么,JNDI到底起什么作用?//帶著問題看文章是最有效的

要了解JNDI的作用,我們可以從“如果不用JNDI我們怎樣做?用了JNDI后我們又將怎樣做?”這個(gè)問題來探討。

沒有JNDI的做法:

程序員開發(fā)時(shí),知道要開發(fā)訪問MySQL數(shù)據(jù)庫的應(yīng)用,于是將一個(gè)對 MySQL JDBC 驅(qū)動(dòng)程序類的引用進(jìn)行了編碼,并通過使用適當(dāng)?shù)?JDBC URL 連接到數(shù)據(jù)庫。
就像以下代碼這樣:

Java代碼 復(fù)制代碼
  1. Connection conn=null;   
  2. try {   
  3.   Class.forName("com.mysql.jdbc.Driver",   
  4.                 true, Thread.currentThread().getContextClassLoader());   
  5.   conn=DriverManager.   
  6.     getConnection("jdbc:mysql://MyDBServer?user=qingfeng&password=mingyue");   
  7.   ......   
  8.   conn.close();   
  9. catch(Exception e) {   
  10.   e.printStackTrace();   
  11. finally {   
  12.   if(conn!=null) {   
  13.     try {   
  14.       conn.close();   
  15.     } catch(SQLException e) {}   
  16.   }   
  17. }  



這是傳統(tǒng)的做法,也是以前非Java程序員(如Delphi、VB等)常見的做法。這種做法一般在小規(guī)模的開發(fā)過程中不會(huì)產(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ū)動(dòng)程序包和類名需要修改;
3、隨著實(shí)際使用終端的增加,原配置的連接池參數(shù)可能需要調(diào)整;
4、......

解決辦法:
程序員應(yīng)該不需要關(guān)心“具體的數(shù)據(jù)庫后臺(tái)是什么?JDBC驅(qū)動(dòng)程序是什么?JDBC URL格式是什么?訪問數(shù)據(jù)庫的用戶名和口令是什么?”等等這些問題,程序員編寫的程序應(yīng)該沒有對 JDBC 驅(qū)動(dòng)程序的引用,沒有服務(wù)器名稱,沒有用戶名稱或口令 —— 甚至沒有數(shù)據(jù)庫池或連接管理。而是把這些問題交給J2EE容器來配置和管理,程序員只需要對這些配置和管理進(jìn)行引用即可。

由此,就有了JNDI。
//看的出來,是為了一個(gè)最最核心的問題:是為了解耦,是為了開發(fā)出更加可維護(hù)、可擴(kuò)展//的系統(tǒng)

用了JNDI之后的做法:
首先,在在J2EE容器中配置JNDI參數(shù),定義一個(gè)數(shù)據(jù)源,也就是JDBC引用參數(shù),給這個(gè)數(shù)據(jù)源設(shè)置一個(gè)名稱;然后,在程序中,通過數(shù)據(jù)源名稱引用數(shù)據(jù)源從而訪問后臺(tái)數(shù)據(jù)庫。

//紅色的字可以看出,JNDI是由j2ee容器提供的功能

具體操作如下(以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ù)庫,如下:
Java代碼 復(fù)制代碼
  1. <?xml version="1.0" encoding="UTF-8"?>   
  2. <datasources>   
  3. <local-tx-datasource>   
  4.     <jndi-name>MySqlDS</jndi-name>   
  5.     <connection-url>jdbc:mysql://localhost:3306/lw</connection-url>   
  6.     <driver-class>com.mysql.jdbc.Driver</driver-class>   
  7.     <user-name>root</user-name>   
  8.     <password>rootpassword</password>   
  9. <exception-sorter-class-name>   
  10. org.jboss.resource.adapter.jdbc.vendor.MySQLExceptionSorter   
  11. </exception-sorter-class-name>   
  12.     <metadata>   
  13.        <type-mapping>mySQL</type-mapping>   
  14.     </metadata>   
  15. </local-tx-datasource>   
  16. </datasources>  


這里,定義了一個(gè)名為MySqlDS的數(shù)據(jù)源,其參數(shù)包括JDBC的URL,驅(qū)動(dòng)類名,用戶名及密碼等。

2、在程序中引用數(shù)據(jù)源:

Java代碼 復(fù)制代碼
  1. Connection conn=null;   
  2. try {   
  3.   Context ctx=new InitialContext();   
  4.   Object datasourceRef=ctx.lookup("java:MySqlDS"); //引用數(shù)據(jù)源   
  5.   DataSource ds=(Datasource)datasourceRef;   
  6.   conn=ds.getConnection();   
  7.   ......   
  8.   c.close();   
  9. catch(Exception e) {   
  10.   e.printStackTrace();   
  11. finally {   
  12.   if(conn!=null) {   
  13.     try {   
  14.       conn.close();   
  15.     } catch(SQLException e) { }   
  16.   }   
  17. }  


直接使用JDBC或者通過JNDI引用數(shù)據(jù)源的編程代碼量相差無幾,但是現(xiàn)在的程序可以不用關(guān)心具體JDBC參數(shù)了。//解藕了,可擴(kuò)展了
在系統(tǒng)部署后,如果數(shù)據(jù)庫的相關(guān)參數(shù)變更,只需要重新配置 mysql-ds.xml 修改其中的JDBC參數(shù),只要保證數(shù)據(jù)源的名稱不變,那么程序源代碼就無需修改。

由此可見,JNDI避免了程序與數(shù)據(jù)庫之間的緊耦合,使應(yīng)用更加易于配置、易于部署。

JNDI的擴(kuò)展:
JNDI在滿足了數(shù)據(jù)源配置的要求的基礎(chǔ)上,還進(jìn)一步擴(kuò)充了作用:所有與系統(tǒng)外部的資源的引用,都可以通過JNDI定義和引用。
//注意什么叫資源

所以,在J2EE規(guī)范中,J2EE 中的資源并不局限于 JDBC 數(shù)據(jù)源。引用的類型有很多,其中包括資源引用(已經(jīng)討論過)、環(huán)境實(shí)體和 EJB 引用。特別是 EJB 引用,它暴露了 JNDI 在 J2EE 中的另外一項(xiàng)關(guān)鍵角色:查找其他應(yīng)用程序組件。

EJB 的 JNDI 引用非常類似于 JDBC 資源的引用。在服務(wù)趨于轉(zhuǎn)換的環(huán)境中,這是一種很有效的方法。可以對應(yīng)用程序架構(gòu)中所得到的所有組件進(jìn)行這類配置管理,從 EJB 組件到 JMS 隊(duì)列和主題,再到簡單配置字符串或其他對象,這可以降低隨時(shí)間的推移服務(wù)變更所產(chǎn)生的維護(hù)成本,同時(shí)還可以簡化部署,減少集成工作。外部資源”。


總結(jié):
J2EE 規(guī)范要求所有 J2EE 容器都要提供 JNDI 規(guī)范的實(shí)現(xiàn)。//sun 果然喜歡制定規(guī)范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)過一些計(jì)劃和預(yù)先考慮,這個(gè)承諾是完全可以實(shí)現(xiàn)的。


從上面的文章中可以看出:
1、JNDI 提出的目的是為了解藕,是為了開發(fā)更加容易維護(hù),容易擴(kuò)展,容易部署的應(yīng)用。
2、JNDI 是一個(gè)sun提出的一個(gè)規(guī)范(類似于jdbc),具體的實(shí)現(xiàn)是各個(gè)j2ee容器提供商,sun   只是要求,j2ee容器必須有JNDI這樣的功能。
3、JNDI 在j2ee系統(tǒng)中的角色是“交換機(jī)”,是J2EE組件在運(yùn)行時(shí)間接地查找其他組件、資源或服務(wù)的通用機(jī)制。
4、JNDI 是通過資源的名字來查找的,資源的名字在整個(gè)j2ee應(yīng)用中(j2ee容器中)是唯一的。

再轉(zhuǎn)一篇文章:


JNDI全稱 Java Naming and Directory Interface
JNDI是Java平臺(tái)的一個(gè)標(biāo)準(zhǔn)擴(kuò)展,提供了一組接口、類和關(guān)于命名空間的概念。如同其它很多Java技術(shù)一樣,JDNI是provider-based的技術(shù),暴露了一個(gè)API和一個(gè)服務(wù)供應(yīng)接口(SPI)。這意味著任何基于名字的技術(shù)都能通過JNDI而提供服務(wù),只要JNDI支持這項(xiàng)技術(shù)。JNDI目前所支持的技術(shù)包括LDAP、CORBA Common Object Service(COS)名字服務(wù)、RMI、NDS、DNS、Windows注冊表等等。很多J2EE技術(shù),包括EJB都依靠JNDI來組織和定位實(shí)體。
JDNI通過綁定的概念將對象和名稱聯(lián)系起來。在一個(gè)文件系統(tǒng)中,文件名被綁定給文件。在DNS中,一個(gè)IP地址綁定一個(gè)URL。在目錄服務(wù)中,一個(gè)對象名被綁定給一個(gè)對象實(shí)體。
JNDI中的一組綁定作為上下文來引用。每個(gè)上下文暴露的一組操作是一致的。例如,每個(gè)上下文提供了一個(gè)查找操作,返回指定名字的相應(yīng)對象。每個(gè)上下文都提供了綁定和撤除綁定名字到某個(gè)對象的操作。JNDI使用通用的方式來暴露命名空間,即使用分層上下文以及使用相同命名語法的子上下文。
jndi的用途:
1。你可以用jndi來得到object類的屬性
如:
Java代碼 復(fù)制代碼
  1. Attribute attr =directory.getAttributes(personName).get("email");    
  2. String email = (String)attr.get();   

2。你可以用jndi來搜索對象
如:
Java代碼 復(fù)制代碼
  1. foxes = directory.search("o=Wiz,c=US""sn=Fox", controls);   

查找誰的名字叫Fox在wiz部門的員工?
3。你可以用jndi通過naming/directory服務(wù)查詢像printers和databases的對象
如:查詢 Printer
Java代碼 復(fù)制代碼
  1. Printer printer = (Printer)namespace.lookup(printerName);    
  2. printer.print(document);   

4。你可以用jndi列表出命名空間的特殊級別的內(nèi)容
如:
Java代碼 復(fù)制代碼
  1. NamingEnumeration list = namespace.list("o=Widget, c=US";    
  2. while (list.hasMore()) {    
  3. NameClassPair entry = (NameClassPair)list.next();    
  4. display(entry.getName(), entry.getClassName());    
  5. }