JNDI

訪問命名和目錄服務的API

JNDI(Java Naming and Directory Interface,Java命名和目錄介面)是SUN公司提供的一種標準的Java命名系統介面,JNDI提供統一的客戶端API,通過不同的訪問提供者介面JNDI服務供應介面(SPI)的實現,由管理者將JNDI API映射為特定的命名服務和目錄系統,使得Java應用程序可以和這些命名服務和目錄服務之間進行交互。目錄服務是命名服務的一種自然擴展。兩者之間的關鍵差別是目錄服務中對象不但可以有名稱還可以有屬性(例如,用戶有email地址),而命名服務中對象沒有屬性。

集群JNDI實現了高可靠性JNDI,通過伺服器的集群,保證了JNDI的負載平衡和錯誤恢復。在全局共享的方式下,集群中的一個應用伺服器保證本地JNDI樹的獨立性,並擁有全局的JNDI樹。每個應用伺服器在把部署的服務對象綁定到自己本地的JNDI樹的同時,還綁定到一個共享的全局JNDI樹,實現全局JNDI和自身JNDI的聯繫。

JNDI(Java Naming and Directory Interface)是一個應用程序設計的API,為開發人員提供了查找和訪問各種命名和目錄服務的通用、統一的介面,類似JDBC都是構建在抽象層上。現在JNDI已經成為J2EE的標準之一,所有的J2EE容器都必須提供一個JNDI的服務。

JNDI可訪問的現有的目錄及服務有:

DNS、XNam、Novell目錄服務、LDAP(Lightweight Directory Access Protocol輕型目錄訪問協議)、CORBA對象服務、文件系統、Windows XP/2000/NT/Me/9x的註冊表、RMI、DSML v1&v2、NIS。

優點


包含了大量的命名和目錄服務,使用通用介面來訪問不同種類的服務;
可以同時連接到多個命名或目錄服務上;
建立起邏輯關聯,允許把名稱同Java對象或資源關聯起來,而不必知道對象或資源的物理ID。
JNDI程序包:
javax.naming:命名操作;
javax.naming.directory:目錄操作;
javax.naming.event:在命名目錄伺服器中請求事件通知;
javax.naming.ldap:提供LDAP支持;
javax.naming.spi:允許動態插入不同實現。
利用JNDI的命名與服務功能來滿足企業級API對命名與服務的訪問,諸如EJB、JMS、JDBC2.0以及IIOP上的RMI通過JNDI來使用CORBA的命名服務。

架構


JNDI架構提供了一組標準的獨立於命名系統的API,這些API構建在與命名系統有關的驅動之上。這一層有助於將應用與實際數據源分離,因此不管應用訪問的是LDAP、RMI、DNS、還是其他的目錄服務。換句話說,JNDI獨立於目錄服務的具體實現,只要有目錄的服務提供介面(或驅動),就可以使用目錄。
關於JNDI要注意的重要一點是,它提供了應用編程介面(applicationprogramminginterface,API)和服務提供者介面(serviceproviderinterface,SPI)。這一點的真正含義是,要讓應用與命名服務或目錄服務交互,必須有這個服務的JNDI服務提供者,這正是JNDISPI發揮作用的地方。服務提供者基本上是一組類,這些類為各種具體的命名和目錄服務實現了JNDI介面—很像JDBC驅動為各種具體的資料庫系統實現了JDBC介面一樣。作為一個應用開發者,不必操心JNDISPI的具體實現。只需要確認要使用的每一個命名或目錄服務都有服務提供者。

組件


1、Javax.naming:包含了訪問命名服務的類和介面。例如,它定義了Context介面,這是命名服務執行查詢的入口。
2、Javax.naming.directory:對命名包的擴充,提供了訪問目錄服務的類和介面。例如,它為屬性增加了新的類,提供了表示目錄上下文的DirContext介面,定義了檢查和更新目錄對象的屬性的方法。
3、Javax.naming.event:提供了對訪問命名和目錄服務時的事件通知的支持。例如,定義了NamingEvent類,這個類用來表示命名/目錄服務產生的事件,定義了偵聽NamingEvents的NamingListener介面。
4、Javax.naming.ldap:這個包提供了對LDAP版本3擴充的操作和控制的支持,通用包javax.naming.directory沒有包含這些操作和控制。
5、Javax.naming.spi:這個包提供了一個方法,通過javax.naming和有關包動態增加對訪問命名和目錄服務的支持。這個包是為有興趣創建服務提供者的開發者提供的。

用途


命名或目錄服務使用戶可以集中存儲共有信息,這一點在網路應用中是重要的,因為這使得這樣的應用更協調、更容易管理。例如,可以將印表機設置存儲在目錄服務中,以便被與印表機有關的應用使用。
大家每天都不知不覺地使用了命名服務。命名系統中的對象可以是DNS記錄中的名稱、應用伺服器中的EJB組件(EnterpriseJavaBeansComponent)、LDAP(LightweightDirectoryAccessProtocol)中的用戶Profile。
目錄服務是命名服務的自然擴展。兩者之間的關鍵差別是目錄服務中對象可以有屬性(例如,用戶有email地址),而命名服務中對象沒有屬性。因此,在目錄服務中,你可以根據屬性搜索對象。JNDI允許你訪問文件系統中的文件,定位遠程RMI註冊的對象,訪問像LDAP這樣的目錄服務,定位網路上的EJB組件。
對於象LDAP客戶端、應用launcher、類瀏覽器、網路管理實用程序,甚至地址薄這樣的應用來說,JNDI是一個很好的選擇。

組成部分


JNDI提供了一種統一的方式,可以用在網路上查找和訪問服務。通過指定一個資源名稱,該名稱對應於資料庫或命名服務中的一個記錄,同時返回資料庫連接建立所必須的信息。
JNDI主要有兩部分組成:應用程序編程介面和服務供應商介面。應用程序編程介面提供了Java應用程序訪問各種命名和目錄服務的功能,服務供應商介面提供了任意一種服務的供應商使用的功能。
代碼示例:try{Contextcntxt=newInitialContext();DataSourceds=(DataSource)cntxt.lookup("jdbc/dpt");catch(NamingExceptionne)

技術應用


消息通信是軟體組件或應用程序用來通信的一種方法。JMS就是一種允許應用程序創建、發送、接收、和讀取消息的JAVA技術。
代碼示例:
try
Propertiesenv=newProperties();
InitialContextinictxt=newInitialContext(env);
TopicConnectionFactoryconnFactory=(TopicConnectionFactory)inictxt.lookup("TTopicConnectionFactory");
catch(NamingExceptionne)
訪問特定目錄:舉個例子,人是個對象,他有好幾個屬性,諸如這個人的姓名、電話號碼、電子郵件地址、郵政編碼等屬性。通過getAttributes()方法
Attributeattr=directory.getAttributes(personName).get("email");
Stringemail=(String)attr.get();
通過使用JNDI讓客戶使用對象的名稱或屬性來查找對象:
foxes=directory.search("o=Wiz,c=US","sn=Fox",controls);
通過使用JNDI來查找諸如印表機、資料庫這樣的對象,查找印表機的例子:
Printerprinter=(Printer)namespace.lookup(printerName);
printer.print(document);
瀏覽命名空間:
NamingEnumerationlist=namespace.list("o=Widget,c=US");
while(list.hasMore())
NameClassPairentry=(NameClassPair)list.next();
display(entry.getName(),entry.getClassName());

常用操作


voidbind(StringsName,Objectobject);――綁定:把名稱同對象關聯的過程
voidrebind(StringsName,Objectobject);――重新綁定:用來把對象同一個已經存在的名稱重新綁定
voidunbind(StringsName);――釋放:用來把對象從目錄中釋放出來
Objectlookup(StringsName);――查找:返回目錄中的一個對象
voidrename(StringsOldName,StringsNewName);――重命名:用來修改對象名稱綁定的名稱
NamingEnumerationlistBinding(StringsName);――清單:返回綁定在特定上下文中對象的清單列表
NamingEnumerationlist(StringsName);
代碼示例:重新得到了名稱、類名和綁定對象。
NamingEnumerationnamEnumList=ctxt.listBinding("cntxtName");
while(namEnumList.hasMore()
Bindingbnd=(Binding)namEnumList.next();
StringsObjName=bnd.getName();
StringsClassName=bnd.getClassName();
SomeObjectobjLocal=(SomeObject)bnd.getObject();