天若有情

          到教堂懺悔,愿主安撫我罪惡的心靈......
          posts - 4, comments - 35, trackbacks - 0, articles - 24

          JNDI全攻略之(一)

          Posted on 2008-09-08 12:27 freedoom 閱讀(306) 評論(0)  編輯  收藏 所屬分類: JAVA技術(好文保存)

          JNDI全攻略之(一)

          關鍵字:JNDI,J2EE,Java,命名和目錄接口,Java Naming and Directory Interface

          摘要:

          本文詳細介紹了JNDI的架構與實現,JNDI的工作原理,并給出了具體代碼,幫助讀者更理解J2EE主要常用技術---JNDI.本文為系列文章的第一篇,其它相關文章會在近期推出。
           
          名詞解釋
              jndi是Java 命名和目錄接口(Java Naming and Directory Interface,JNDI)的簡稱.從一開始就一直是 Java 2 平臺企業版(JEE)的核心技術之一。在JMS,JMail,JDBC,EJB等技術中,就大量應用的這種技術。
              
          為什么會有jndi
              jndi誕生的理由似乎很簡單。隨著分布式應用的發展,遠程訪問對象訪問成為常用的方法。雖然說通過Socket等編程手段仍然可實現遠程通信,但按照模式的理論來說,仍是有其局限性的。RMI技術,RMI-IIOP技術的產生,使遠程對象的查找成為了技術焦點。JNDI技術就應運而生。JNDI技術產生后,就可方便的查找遠程或是本地對象。
           
          JNDI的架構與實現
              
          JNDI的架構與JDBC的架構非常類似.JNDI架構提供了一組標準命名系統的API,這些API在JDK1.3之前是作為一個單獨的擴展包jndi.jar(通過這個地址下載),這個基礎API構建在與SPI之上。這個API提供如下五個包
          在應用程序中,我們實際上只使到用以上幾個包的中類.具體調用類及通信過程對用戶來說是透明的.
          JNDI API提供了訪問不同JNDI服務的一個標準的統一的實現,其具體實現可由不同的 Service Provider來完成。前面講的為第一層JNDI API層.
           
          最下層為JNDI SPI API及其具體實現。
           

          圖中所列的一些SPI可從http://java.sun.com/products/jndi/downloads/index.html下載.


          包括了幾個增強和下面的命名/目錄服務提供者:
          • LDAP(Lightweight Directory Access Protocol)服務提供者
          • CORBA COS(Common Object Request Broker Architecture Common Object Services)命名服務提供者 
          • RMI(Java Remote Method Invocation)注冊服務提供者
          • DNS(Domain Name System)服務提供者.
          • FSSP(File System Service Provider)文件系統服務提供者
          • 其它服務提供者
           
          中間層為命名管理層。其功能應該由JNDI SPI來完成。上層為JNDI API,這個API包在Java 2 SDK 1.3及以上的版本中已經包括。
           
          前面講解的只是作為應用程序客戶端的架構實現,其服務端是由SPI對應的公司/廠商來實現的,我們只需將服務端的相關參數傳給JNDI API就可以了,具體調用過程由SPI來完成.
           
           
          JNDI工作原理
           

          下面通過一個示例程序來說明JNDI工作原理(代碼為自解釋).

          /*
          * Created on 2005-3-4
          *
          * To change the template for this generated file go to
          * Window>Preferences>Java>Code Generation>Code and Comments
          */
          package com.sily.jndi;
          import java.io.FileInputStream;
          import java.util.Properties;
          import javax.naming.Context;
          import javax.naming.InitialContext;
          /**
          * @author shizy
          *
          * To change the template for this generated type comment go to
          * Window>Preferences>Java>Code Generation>Code and Comments
          */
          public class TestJbossJNDI {
          /**
          *
          */
          public TestJbossJNDI() {
          super();
          // TODO Auto-generated constructor stub
          }
          public static void main(String[] args) { try {
          Properties env = new Properties();
          //載入jboss的SPI相關參數,包括初始上下文工廠,服務URL,等等
          env.load(new FileInputStream("jbossJndi.properties"));
          env.list(System.out);
          //通過JNDI api 初始化上下文
          InitialContext ctx = new javax.naming.InitialContext(env);
          System.out.println("Got context");
          //create a subContext
          ctx.createSubcontext("/sylilzy");
          ctx.createSubcontext("sylilzy/sily");
          //rebind a object
          ctx.rebind("sylilzy/sily/a", "I am sily a!");
          ctx.rebind("sylilzy/sily/b", "I am sily b!");
          //lookup context
          Context ctx1=(Context)ctx.lookup("sylilzy");
          Context ctx2=(Context)ctx1.lookup("/sylilzy/sily");
          ctx2.bind("/sylilzy/g", "this is g");
          //lookup binded object
          Object o;
          o=ctx1.lookup("sily/a");
          System.out.println("get object from jndi:"+o);
          //rename the object
          ctx2.rename("/sylilzy/g", "g1");
          o=ctx2.lookup("g1");
          System.out.println("get object from jndi:"+o);
          } catch (Exception e) {
          e.printStackTrace();
          }
          }
          }
          結果輸出如下:
          -- listing properties --
          java.naming.factory.initial=org.jnp.interfaces.NamingContextFactory
          java.naming.provider.url=jnp://localhost:1099
          java.naming.factory.url.pkgs=org.jboss.naming:org.jnp.interfaces
          Got context
          get object from jndi:I am sily a!
          get object from jndi:this is g
           
          程序中jbossJndi.properties文件的內容為:
          java.naming.factory.initial=org.jnp.interfaces.NamingContextFactory
          java.naming.factory.url.pkgs=org.jboss.naming:org.jnp.interfaces
          java.naming.provider.url=jnp://localhost:1099
           
          注意:要正確運行示例程序,請啟動jboss,并將jboss的jbossall-client.jar文件放入classpath中。
           
          上述示例程序在jboss服務器的jndi樹上建立了幾個上下文,并bind了幾對象,大家可通過附錄中的代碼或其它工具查看
          查看結果為:
          -----------------------------
          /sylilzy/sily
          -----------------------------
          /sylilzy/sily/b:I am sily b!
          /sylilzy/sily/a:I am sily a!
          /sylilzy/sily/g1:this is g
          -----------------------------
          -----------------------------
           
          上述程序中,我們的代碼只涉及到了jndi API,其它細節如初始化jboss jndi的初始上下文,建立網絡連接,與服務器通信,對我們來說都是透明的,另外,我們將jboss jndi的spi包中的類名作為參數傳入了程序中,要訪問一個遠程對象,我們所做的就這么多。
           
          下面,再提供一個例子,與上例不同,我們不需要jboss,我們使用sun的FSSP(File System Service Provider)文件系統服務提供者.注意在這個例子中要使用到前面所說的File System Service Provider for the Java Naming and Directory InterfaceTM (JNDI)相關類(下載)。
          /*
          * Created on 2005-3-1
          *
          * To change the template for this generated file go to
          * Window>Preferences>Java>Code Generation>Code and Comments
          */
          package com.sily.jndi;
          import java.io.FileInputStream;
          import java.util.Properties;
          import javax.naming.*;
          import javax.naming.Context;
          import javax.naming.InitialContext;
          /**
          * @author shizy
          *
          * To change the template for this generated type comment go to
          * Window>Preferences>Java>Code Generation>Code and Comments
          */
          public class JndiTest1 {
          /**
          *
          */
          public JndiTest1() {
            super();
            // TODO Auto-generated constructor stub
           }
           public static void main(String[] args) {
            try {
             Properties env = new Properties();
             env.load(new FileInputStream("fileSystemService.properties"));
             env.put(Context.PROVIDER_URL, "file:///c:/");
             Context ctx = new InitialContext(env);
             ctx.createSubcontext("sylilzy");
          
          NamingEnumeration list = ctx.list("/");
             while (list.hasMore()) {
              NameClassPair nc = (NameClassPair) list.next();
              System.out.println(nc);
             }
          
          }
            catch (Exception e) {
             e.printStackTrace();
            }
           }
          }
          
           
          上例中fileSystemService.properties文件的內容為:java.naming.factory.initial=com.sun.jndi.fscontext.RefFSContextFactory
           
          這個例子較簡單,運行后,它會列出C:\下所有的文件和目錄,另外你會發現有一個新目錄被創建了.本例不同于上例,它并不需要服務端,因為它訪問的是文件系統.有關幫助可查閱包內的相關文檔。
           
          通過對比這兩個例子,應該JNDI的工作原理有了一個大致的了解。
          總結:
          jndi技術體現了分布式應用的優點,同進它的產生也為分布式對象提供了統一的訪問接口。由于篇幅所限,對目錄的操作本文未作介紹,其它內容將在接下來的系列中討論。要對JNDI技術作全面的了解,請參閱參考資料.要對于JNDI技術深入學習,仍有許多地方值得進一步了解,例如EJB容器所使用的JNDI所提供的對象就有 Local和Remote之分,對于Local Object,對于不同的JVM是不可訪問的;對于遠程對象的訪問,還涉及到Java安全機制。
           
          附錄:
          查看jboss jndi內容的代碼:
          //----------------------------------------
          /*
          * Created on 2005-3-4
          *
          * To change the template for this generated file go to
          * Window>Preferences>Java>Code Generation>Code and Comments
          */
          package com.sily.jndi;
          import java.io.FileInputStream;
          import java.util.Properties;
          import javax.naming.*;
          import javax.naming.Context;
          import javax.naming.InitialContext;
          /**
          * @author shizy
          *
          * To change the template for this generated type comment go to
          * Window>Preferences>Java>Code Generation>Code and Comments
          */
          public class ListJbossJndi {
          /**
          *
          */
          public ListJbossJndi() {
            super();
            // TODO Auto-generated constructor stub
           }
           public static void main(String[] args) {
            try {
             Properties env = new Properties();
             env.load(new FileInputStream("jbossJndi.properties"));
             //env.list(System.out);
             Context ctx = new InitialContext(env);
             listCtx(ctx.lookup("sylilzy"));
            }
            catch (Exception e) {
             e.printStackTrace();
            }
           }
           static void listCtx(Object o){
            if(!(o instanceof Context))log(":"+o);
            else {
             log("\n-----------------------------");
            try {
             Context ctx=(Context)o;
             //log(ctx.getNameInNamespace()+"/:");
             NamingEnumeration list=ctx.listBindings("");
             while(list.hasMore()){
              Binding bind=(Binding)list.next();
              log("\n/"+ctx.getNameInNamespace()+"/"+bind.getName());
              listCtx(bind.getObject());
             }
             log("\n-----------------------------");
            }
            catch (NamingException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
            }
           }
           }
           static void log(Object o){
            System.out.print(o);
           }
          }
          
          
              
           

          作者簡介:
              施祖陽,網名sylilzy,1979年生。
              2002年起從事軟件開發工作,主要研究為JAVA、Linux及相關技術。
              你可通過sylilzy@163.com與作者聯系。
           
          參考資料:
           
          主站蜘蛛池模板: 凭祥市| 得荣县| 永胜县| 石棉县| 新野县| 潜江市| 朔州市| 永州市| 鄯善县| 通州市| 咸宁市| 宾阳县| 和静县| 塔城市| 潞城市| 卢龙县| 朝阳区| 积石山| 杂多县| 宜宾市| 崇义县| 克东县| 晋中市| 鹤庆县| 醴陵市| 襄城县| 突泉县| 临夏县| 淮滨县| 股票| 贵州省| 诸城市| 噶尔县| 霍山县| 丰顺县| 图木舒克市| 伊金霍洛旗| 台江县| 泸定县| 郁南县| 湄潭县|