JNDI的理解
JNDI是 Java 命名與文件夾接口(Java Naming and Directory Interface),在J2EE規(guī)范中是重要的規(guī)范之中的一個(gè),不少專家覺(jué)得,沒(méi)有透徹理解JNDI的意義和作用,就沒(méi)有真正掌握J(rèn)2EE特別是EJB的知識(shí)。
那么,JNDI究竟起什么作用?//帶著問(wèn)題看文章是最有效的
要了解JNDI的作用,我們能夠從“假設(shè)不用JNDI我們?nèi)绾巫觯坑昧薐NDI后我們又將如何做?”這個(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)程序類的引用進(jìn)行了編碼,并通過(guò)使用適當(dāng)?shù)?JDBC URL 連接到數(shù)據(jù)庫(kù)。
就像以下代碼這樣:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
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.close(); } catch (Exception e) { e.printStackTrace(); } finally { if (conn!= null ) { try { conn.close(); } catch (SQLException e) {} } } |
這是傳統(tǒng)的做法,也是曾經(jīng)非Java程序猿(如Delphi、VB等)常見(jiàn)的做法。
這種做法一般在小規(guī)模的開(kāi)發(fā)過(guò)程中不會(huì)產(chǎn)生問(wèn)題,僅僅要程序猿熟悉Java語(yǔ)言、了解JDBC技術(shù)和MySQL,能夠非常快開(kāi)發(fā)出對(duì)應(yīng)的應(yīng)用程序。
沒(méi)有JNDI的做法存在的問(wèn)題:
- 數(shù)據(jù)庫(kù)server名稱MyDBServer 、username和口令都可能須要改變,由此引發(fā)JDBC URL須要改動(dòng);
- 數(shù)據(jù)庫(kù)可能改用別的產(chǎn)品,如改用DB2或者Oracle,引發(fā)JDBC驅(qū)動(dòng)程序包和類名須要改動(dòng);
- 隨著實(shí)際使用終端的添加,原配置的連接池參數(shù)可能須要調(diào)整;
- ......
解決的方法:
程序猿應(yīng)該不須要關(guān)心“詳細(xì)的數(shù)據(jù)庫(kù)后臺(tái)是什么?JDBC驅(qū)動(dòng)程序是什么?JDBC URL格式是什么?訪問(wèn)數(shù)據(jù)庫(kù)的username和口令是什么?”等等這些問(wèn)題。程序猿編寫(xiě)的程序應(yīng)該沒(méi)有對(duì) JDBC 驅(qū)動(dòng)程序的引用,沒(méi)有server名稱,沒(méi)實(shí)username稱或口令 —— 甚至沒(méi)有數(shù)據(jù)庫(kù)池或連接管理。
而是把這些問(wèn)題交給J2EE容器(比方weblogic)來(lái)配置和管理,程序猿僅僅須要對(duì)這些配置和管理進(jìn)行引用就可以。
由此,就有了JNDI。
//看的出來(lái)。是為了一個(gè)最最核心的問(wèn)題:是為了解耦,是為了開(kāi)發(fā)出更加可維護(hù)、可擴(kuò)展//的系統(tǒng)
用了JNDI之后的做法:
首先。在在J2EE容器中配置JNDI參數(shù),定義一個(gè)數(shù)據(jù)源。也就是JDBC引用參數(shù),給這個(gè)數(shù)據(jù)源設(shè)置一個(gè)名稱;然后,在程序中,通過(guò)數(shù)據(jù)源名稱引用數(shù)據(jù)源從而訪問(wèn)后臺(tái)數(shù)據(jù)庫(kù)。
//紅色的字能夠看出。JNDI是由j2ee容器提供的功能
詳細(xì)操作例如以下(以JBoss為例):
1、配置數(shù)據(jù)源
在JBoss 的 D:\jboss420GA\docs\examples\jca 文件夾以下。有非常多不同數(shù)據(jù)庫(kù)引用的數(shù)據(jù)源定義模板。
將當(dāng)中的 mysql-ds.xml 文件Copy到你使用的server下,如 D:\jboss420GA\server\default\deploy。
改動(dòng) mysql-ds.xml 文件的內(nèi)容,使之能通過(guò)JDBC正確訪問(wèn)你的MySQL數(shù)據(jù)庫(kù)。例如以下:
Java代碼
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
<? 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)類名,username及密碼等。
2、在程序中引用數(shù)據(jù)源:
Java代碼
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
Connection conn= null ; try { Context ctx= new InitialContext(); Object datasourceRef=ctx.lookup( "java:MySqlDS" ); //引用數(shù)據(jù)源 DataSource ds=(Datasource)datasourceRef; conn=ds.getConnection(); ...... c.close(); } catch (Exception e) { e.printStackTrace(); } finally { if (conn!= null ) { try { conn.close(); } catch (SQLException e) { } } } |
直接使用JDBC或者通過(guò)JNDI引用數(shù)據(jù)源的編程代碼量相差無(wú)幾,可是如今的程序能夠不用關(guān)心詳細(xì)JDBC參數(shù)了。
//解藕了。可擴(kuò)展了
在系統(tǒng)部署后。假設(shè)數(shù)據(jù)庫(kù)的相關(guān)參數(shù)變更。僅僅須要又一次配置 mysql-ds.xml 改動(dòng)當(dāng)中的JDBC參數(shù),僅僅要保證數(shù)據(jù)源的名稱不變,那么程序源碼就無(wú)需改動(dòng)。
由此可見(jiàn)。JNDI避免了程序與數(shù)據(jù)庫(kù)之間的緊耦合,使應(yīng)用更加易于配置、易于部署。
JNDI的擴(kuò)展:
JNDI在滿足了數(shù)據(jù)源配置的要求的基礎(chǔ)上。還進(jìn)一步擴(kuò)充了作用:全部與系統(tǒng)外部的資源的引用,都能夠通過(guò)JNDI定義和引用。
//注意什么叫資源
所以,在J2EE規(guī)范中,J2EE 中的資源并不局限于 JDBC 數(shù)據(jù)源。
引用的類型有非常多,當(dāng)中包含資源引用(已經(jīng)討論過(guò))、環(huán)境實(shí)體和 EJB 引用。
特別是 EJB 引用,它暴露了 JNDI 在 J2EE 中的另外一項(xiàng)關(guān)鍵角色:查找其它應(yīng)用程序組件。
EJB 的 JNDI 引用非常相似于 JDBC 資源的引用。在服務(wù)趨于轉(zhuǎn)換的環(huán)境中,這是一種非常有效的方法。能夠?qū)?yīng)用程序架構(gòu)中所得到的全部組件進(jìn)行這類配置管理,從 EJB 組件到 JMS 隊(duì)列和主題。再到簡(jiǎn)單配置字符串或其它對(duì)象。這能夠降低隨時(shí)間的推移服務(wù)變更所產(chǎn)生的維護(hù)成本,同一時(shí)候還能夠簡(jiǎn)化部署,降低集成工作。外部資源”。
總結(jié):
J2EE 規(guī)范要求全部 J2EE 容器都要提供 JNDI 規(guī)范的實(shí)現(xiàn)。//sun 果然喜歡制定規(guī)范JNDI 在 J2EE 中的角色就是“交換機(jī)” —— J2EE 組件在執(zhí)行時(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è)承諾是全然能夠?qū)崿F(xiàn)的。
從上面的文章中能夠看出:
- JNDI 提出的目的是為了解藕,是為了開(kāi)發(fā)更加easy維護(hù),easy擴(kuò)展。easy部署的應(yīng)用。
- JNDI 是一個(gè)sun提出的一個(gè)規(guī)范(相似于jdbc),詳細(xì)的實(shí)現(xiàn)是各個(gè)j2ee容器提供商。sun 僅僅是要求,j2ee容器必須有JNDI這種功能。
- JNDI 在j2ee系統(tǒng)中的角色是“交換機(jī)”,是J2EE組件在執(zhí)行時(shí)間接地查找其它組件、資源或服務(wù)的通用機(jī)制。
- JNDI 是通過(guò)資源的名字來(lái)查找的,資源的名字在整個(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ù)都能通過(guò)JNDI而提供服務(wù)。僅僅要JNDI支持這項(xiàng)技術(shù)。
JNDI眼下所支持的技術(shù)包含LDAP、CORBA Common Object Service(COS)名字服務(wù)、RMI、NDS、DNS、Windows注冊(cè)表等等。非常多J2EE技術(shù),包含EJB都依靠JNDI來(lái)組織和定位實(shí)體。
JDNI通過(guò)綁定的概念將對(duì)象和名稱聯(lián)系起來(lái)。
在一個(gè)文件系統(tǒng)中。文件名稱被綁定給文件。在DNS中,一個(gè)IP地址綁定一個(gè)URL。在文件夾服務(wù)中。一個(gè)對(duì)象名被綁定給一個(gè)對(duì)象實(shí)體。
JNDI中的一組綁定作為上下文來(lái)引用。每一個(gè)上下文暴露的一組操作是一致的。比如,每一個(gè)上下文提供了一個(gè)查找操作。返回指定名字的對(duì)應(yīng)對(duì)象。每一個(gè)上下文都提供了綁定和撤除綁定名字到某個(gè)對(duì)象的操作。
JNDI使用通用的方式來(lái)暴露命名空間,即使用分層上下文以及使用同樣命名語(yǔ)法的子上下文。
jndi的用途:
1。你能夠用jndi來(lái)得到object類的屬性
如:
Java代碼
1
2
|
Attribute attr =directory.getAttributes(personName).get( "email" ); String email = (String)attr.get(); |
2。你能夠用jndi來(lái)搜索對(duì)象
如:
Java代碼
1
|
foxes = directory.search( "o=Wiz,c=US" , "sn=Fox" , controls); |
查找誰(shuí)的名字叫Fox在wiz部門(mén)的員工?
3。你能夠用jndi通過(guò)naming/directory服務(wù)查詢像printers和databases的對(duì)象
如:查詢 Printer
Java代碼
1
2
|
Printer printer = (Printer)namespace.lookup(printerName); printer.print(document); |
4。你能夠用jndi列表出命名空間的特殊級(jí)別的內(nèi)容
如:
Java代碼
1
2
3
4
5
|
NamingEnumeration list = namespace.list( "o=Widget, c=US" ; while (list.hasMore()) { NameClassPair entry = (NameClassPair)list.next(); display(entry.getName(), entry.getClassName()); } |
到此這篇關(guān)于Java JNDI案例詳解的文章就介紹到這了,更多相關(guān)Java JNDI內(nèi)容請(qǐng)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://www.cnblogs.com/zhchoutai/p/7389089.html