JNDI是什么,怎么理解


JNDI 是什么

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

那么,JNDI到底起什么作用?

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

沒有JNDI的做法:

程序員開發時,知道要開發訪問MySQL數據庫的應用,於是將一個對 MySQL JDBC 驅動程序類的引用進行了編碼,並通過使用適當的 JDBC URL 連接到數據庫。
就像以下代碼這樣:

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) {}  
 }  
}

  

這是傳統的做法,也是以前非Java程序員(如Delphi、VB等)常見的做法。這種做法一般在小規模的開發過程中不會產生問題,只要程序員熟悉Java語言、了解JDBC技術和MySQL,可以很快開發出相應的應用程序。

沒有JNDI的做法存在的問題:

1、數據庫服務器名稱MyDBServer 、用戶名和口令都可能需要改變,由此引發JDBC URL需要修改;

2、數據庫可能改用別的產品,如改用DB2或者Oracle,引發JDBC驅動程序包和類名需要修改;

3、隨着實際使用終端的增加,原配置的連接池參數可能需要調整;

4、......

解決辦法:

程序員應該不需要關心“具體的數據庫后台是什么?JDBC驅動程序是什么?JDBC URL格式是什么?訪問數據庫的用戶名和口令是什么?”等等這些問題,程序員編寫的程序應該沒有對 JDBC 驅動程序的引用,沒有服務器名稱,沒有用戶名稱或口令 —— 甚至沒有數據庫池或連接管理。而是把這些問題交給J2EE容器來配置和管理,程序員只需要對這些配置和管理進行引用即可。

由此,就有了JNDI。

用了JNDI之后的做法:

首先,在在J2EE容器中配置JNDI參數,定義一個數據源,也就是JDBC引用參數,給這個數據源設置一個名稱;然后,在程序中,通過數據源名稱引用數據源從而訪問后台數據庫。

具體操作如下(以JBoss為例):

1、配置數據源

在JBoss的 D:/jboss420GA/docs/examples/jca 文件夾下面,有很多不同數據庫引用的數據源定義模板。將其中的 mysql-ds.xml 文件Copy到你使用的服務器下,如 D:/jboss420GA/server/default/deploy。

修改 mysql-ds.xml 文件的內容,使之能通過JDBC正確訪問你的MySQL數據庫,如下:

 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>org.jboss.resource.adapter.jdbc.vendor.MySQLExceptionSorter</exception-sorter-class-name>
10     <metadata>
11        <type-mapping>mySQL</type-mapping>
12     </metadata>
13 </local-tx-datasource>
14 </datasources>

這里,定義了一個名為MySqlDS的數據源,其參數包括JDBC的URL,驅動類名,用戶名及密碼等。

2、在程序中引用數據源:

 

Connection conn=null;
try {
  Context ctx=new InitialContext();
  Object datasourceRef=ctx.lookup("java:MySqlDS"); //引用數據源
  DataSource ds=(Datasource)datasourceRef;
  conn=ds.getConnection();
  /* 使用conn進行數據庫SQL操作 */
  ......
  c.close();
} 
catch(Exception e) {
  e.printStackTrace();
} 
finally {
  if(conn!=null) {
    try {
      conn.close();
    } catch(SQLException e) { }
  }
}

 

直接使用JDBC或者通過JNDI引用數據源的編程代碼量相差無幾,但是現在的程序可以不用關心具體JDBC參數了。
在系統部署后,如果數據庫的相關參數變更,只需要重新配置 mysql-ds.xml 修改其中的JDBC參數,只要保證數據源的名稱不變,那么程序源代碼就無需修改。

由此可見,JNDI避免了程序與數據庫之間的緊耦合,使應用更加易於配置、易於部署。

JNDI的擴展:JNDI在滿足了數據源配置的要求的基礎上,還進一步擴充了作用:所有與系統外部的資源的引用,都可以通過JNDI定義和引用。

所以,在J2EE規范中,J2EE 中的資源並不局限於 JDBC 數據源。引用的類型有很多,其中包括資源引用(已經討論過)、環境實體和 EJB 引用。特別是 EJB 引用,它暴露了 JNDI 在 J2EE 中的另外一項關鍵角色:查找其他應用程序組件。

EJB 的 JNDI 引用非常類似於 JDBC 資源的引用。在服務趨於轉換的環境中,這是一種很有效的方法。可以對應用程序架構中所得到的所有組件進行這類配置管理,從 EJB 組件到 JMS 隊列和主題,再到簡單配置字符串或其他對象,這可以降低隨時間的推移服務變更所產生的維護成本,同時還可以簡化部署,減少集成工作。 外部資源”。

總 結:

J2EE 規范要求所有 J2EE 容器都要提供 JNDI 規范的實現。JNDI 在 J2EE 中的角色就是“交換機” —— J2EE 組件在運行時間接地查找其他組件、資源或服務的通用機制。在多數情況下,提供 JNDI 供應者的容器可以充當有限的數據存儲,這樣管理員就可以設置應用程序的執行屬性,並讓其他應用程序引用這些屬性(Java 管理擴展(Java Management Extensions,JMX)也可以用作這個目的)。JNDI 在 J2EE 應用程序中的主要角色就是提供間接層,這樣組件就可以發現所需要的資源,而不用了解這些間接性。

在 J2EE 中,JNDI 是把 J2EE 應用程序合在一起的粘合劑,JNDI 提供的間接尋址允許跨企業交付可伸縮的、功能強大且很靈活的應用程序。這是 J2EE 的承諾,而且經過一些計划和預先考慮,這個承諾是完全可以實現的。

最近一直在對J2EE的筆記進行整理和復習,雖然J2EE視頻是看過一遍了,但是當我看自己做的筆記的時候陌生程度還是很大,而真正的對某個概念有所認識的時候是將筆記和以前看過的視頻印象進行摩擦,J2EE主要講解的內容是各個規范,再清楚一些就是各個概念,現階段的目標並不是掌握J2EE,而是對J2EE進行輪廓和概念上的了解和認識,到下一步DRP項目中再深層次的對各個規范進行摩擦和認識。

JNDI,翻譯為Java命名和目錄結構(JavaNaming And Directory Interface)官方對其解釋為JNDI是一組在Java應用中訪問命名和目錄服務的API(ApplicationProgramming Interface)說明很精煉,但是比較抽象。

上面的解釋中提高了命名服務和目錄服務兩個概念.先要了解JNDI就必須知道,命名服務和目錄服務是做什么用的。

學習新的概念和知識,比較有效的方式是通過和以前所學過的內容進行聯系,比較。

關於命名服務,其實我們很多時候都在用它,但是並不知道它是它,比較典型的是域名服務器DNS(Domain Naming Service),大對人對DNS還是比較了解的,它是將域名映射到IP地址的服務.比如百度的域名www.baidu.com所映射的IP地址是http://202.108.22.5/,你在瀏覽器中輸入兩個內容是到的同一個頁面.用命名服務器的原因是因為我們記憶baidu這幾個有意義的字母要比記202.108.22.5更容易記憶,但如果站到計算機的角度上,它更喜歡處理這些數字。

從我們生活中找的話還有很多類似的例子,比如說你的身份證號和你的名字可以"理解"成一種命名服務,你的學號和姓名也可以"解釋"為一種命名服務。

可以看出命名服務的特點:一個值和另一個值的映射,將我們人類更容易認識的值同計算機更容易認識的值進行一一映射。

到現在應該對命名服務有所理解吧?

至於目錄服務,從計算機角度理解為在互聯網上有着各種各樣的資源和主機,但是這些內容都是散落在互聯網中,為了訪問這些散落的資源並獲得相應的服務,就需要用到目錄服務。

從我們日常生活中去理解目錄服務的概念可以從電話簿說起,電話簿本身就是一個比較典型的目錄服務,如果你要找到某個人的電話號碼,你需要從電話簿里找到這個人的名稱,然后再看其電話號碼。

 

理解了命名服務和目錄服務再回過頭來看JDNI,它是一個為Java應用程序提供命名服務的應用程序接口,為我們提供了查找和訪問各種命名和目錄服務的通用統一的接口.通過JNDI統一接口我們可以來訪問各種不同類型的服務.如下圖所示,我們可以通過JNDI API來訪問剛才談到的DNS。

 

來源:zhaosg198312


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM