對java:comp/env的研究

這兩天研究了一下 context.lookup("java:comp/env/XXX")和直接context.lookup("XXX")的區別 

網上關於這兩個的文章也不少,可是都說得很難理解,好比什麼ENC環境啊什麼的,各類概念。 

其實說得簡單點:context.lookup("java:comp/env/XXX")只能用在J2EE環境,便是若是你本身寫一個main函數,想經過context.lookup("java:comp/env/XXX")這樣的方式來訪問JNDI服務,這是不可能的。 
由於:java:comp/env/是一個J2EE環境的定義,說白了就是表明當前J2EE應用的環境,好比你本身項目的Web環境或者是EJB環境,那是否是隻要是個Web項目,就能用context.lookup("java:comp/env/XXX")這種方式訪問JNDI服務了呢?也不是!!! 
使用這樣的方式必須作一次 當前應用環境 到 資源名 的映射。 
在web.xml文件中有這樣的標籤: 
<resource-env-ref></resource-env-ref> 
<resource-ref></resource-ref> 
<ejb-local-ref></ejb-local-ref>  
<ejb-ref></ejb-ref> 
這些標籤就是用來創建當前應用環境到服務器資源的映射的。 
有了這樣的映射以後,就能採用context.lookup("java:comp/env/XXX")的方式來訪問JNDI資源了。 


注意:context.lookup("XXX")在任什麼時候候都是有效的,只要XXX確實是一個存在的JNDI名。 


舉個例子: 
用weblogic10的控制檯定了了一個oracle數據源,這個數據源的JNDI名稱是:adsl,那麼只要正確鏈接上了weblogic(固然需傳遞URL,用戶名和密碼還有weblogic的JNDI工廠對象,這不屬於該文的討論範圍)在任何地方都能用context.lookup("xxx")獲得這個數據源了,可是想經過context.lookup("java:comp/env/jdbc/adsl")訪問到這個數據源,就不行了。若是想這樣寫的話須要作兩件事情: 
一、確保你的調用程序是一個web項目或者EJB項目,並部署到weblogic上。(例子使用web項目) 
二、在web項目的web.xml裏面加上以下配置: 
<resource-ref> 
  <res-ref-name>jdbc/adsl</res-ref-name> 
  <res-auth>Container</res-auth> 
  <mapped-name>adsl</mapped-name> <!-- 這個必須和你的全局JNDI數據源名稱同樣 -->  
</resource-ref> 
那麼你就能在你的web程序裏經過context.lookup("java:comp/env/jdbc/adsl")訪問到這個數據源了。 

關於EJB的訪問也相似這樣,能夠直接訪問全局EJB的JNDI名,和能夠映射以後從java:comp/env/下進行訪問。 


以上的講解有錯。。。。。 
上面的狀況只適用於weblogic和J2EE 2.5的規範下。 
一:若是不是J2EE 2.5的規範:那麼在你的web項目的web.xml內將根本不會存在<mapped-name>這個標籤,那麼怎樣才能映射全局JNDI資源到你的項目呢? 
以weblogic爲例,你須要在和web.xml同級的目錄下創建:weblogic.xml文件,在該文件裏面寫上: 
<weblogic-web-app> 
<resource-description> 
<res-ref-name>jdbc/adsl</res-ref-name> 
<jndi-name>adsl</jndi-name><!-- 這個就是全局JNDI資源名 --> 
</resource-description> 
</weblogic-web-app> 

而後你仍然須要在web.xml裏面配置上: 
<resource-ref> 
  <res-ref-name>jdbc/adsl</res-ref-name> 
  <res-auth>Container</res-auth> 
  </resource-ref> 
這樣你才能夠在你的程序裏面經過context.lookup("java:comp/env/jdbc/adsl")訪問到這個數據源。 
這個是weblogic.xml的官方幫助文檔:http://edocs.weblogicfans.net/wls/docs92/webapp/weblogic_xml.html 

二:若是不是使用weblogic,可參見這篇文章:http://blog.csdn.net/lovingprince/article/details/6577920 
其實各類不一樣的J2EE容器,都用不一樣的配置方式,以我目前的測試來看,weblogic彷佛不能配置「私有的JNDI資源」,就是不能配置本身項目單獨的JNDI資源(這只是個人觀點,我目前沒發現怎麼在weblogic裏面配置私有JNDI資源) 
可是研究tomcat6以後發現,tomcat能夠配置 全局JNDI 和私有JNDI(注意這裏說的Tomcat6): 
!!!首先想要說明的是Tomcat的配置不須要修改web.xml裏面的任何內容!!!!! 
!!!Tomcat的全局JNDI資源不能直接訪問,必須有java:comp/env/前綴!!!!!! 
    全局的JNDI配置在server.xml裏面的<GlobalNamingResources>標籤裏面添加以下配置: 
    <Resource name="jdbc/test" 
  auth="Container" 
              type="javax.sql.DataSource" 
              driverClassName="com.mysql.jdbc.Driver" 
              url="jdbc:mysql://127.0.0.1/test" 
              username="root" 
              password="root" 
              maxActive="20" 
              maxIdle="10" 
              maxWait="-1"/> 
    而後某一個項目想要引用這個全局的JNDI,就須要在項目的META-INF下面創建context.xml文件,在裏面寫上: 
<?xml version="1.0" encoding="UTF-8"?> 
<Context> 
    <ResourceLink name="jdbc/test" global="jdbc/test" type="javax.sql.DataSource"/> 
</Context> 
這樣就能夠在程序裏面經過context.lookup("java:comp/env/jdbc/test")進行訪問了。 

    私有的JNDI有三種方式能夠配置: 
      一、能夠直接在server.xml裏面的<Host>節點下添加以下配置: 
<Context path="/test_tomcat6_jndi"> 
<Resource name="jdbc/test" 
      auth="Container" 
              type="javax.sql.DataSource" 
              driverClassName="com.mysql.jdbc.Driver" 
              url="jdbc:mysql://127.0.0.1/test" 
              username="root" 
              password="root" 
              maxActive="20" 
              maxIdle="10" 
              maxWait="-1"/> 
</Context> 
這樣就能夠直接在程序中經過context.lookup("java:comp/env/jdbc/test")訪問了,須要注意的是path="/test_tomcat6_jndi",這個名字必須和你的項目名稱相同,並且不能少了那個斜槓,並且你的項目是經過拷貝文件夾到webapps下面的方式進行的部署。 

     二、 也能夠在conf/context.xml裏面增長以下配置: 
     <Resource name="jdbc/test" 
      auth="Container" 
              type="javax.sql.DataSource" 
              driverClassName="com.mysql.jdbc.Driver" 
              url="jdbc:mysql://127.0.0.1/test" 
              username="root" 
              password="root" 
              maxActive="20" 
              maxIdle="10" 
              maxWait="-1"/> 
這樣就能夠直接在程序中經過context.lookup("java:comp/env/jdbc/test")訪問了 

      三、還能夠在項目的WebRoot下面的META-INF文件夾下面建立context.xml文件,再在context.xml文件裏面寫上 
<?xml version="1.0" encoding="UTF-8"?> 
<Context> 
    <Resource name="jdbc/test" 
  auth="Container" 
              type="javax.sql.DataSource" 
              driverClassName="com.mysql.jdbc.Driver" 
              url="jdbc:mysql://127.0.0.1/test" 
              username="root" 
              password="root" 
              maxActive="20" 
              maxIdle="10" 
              maxWait="-1"/> 
</Context> 
這樣就能夠直接在程序中經過context.lookup("java:comp/env/jdbc/test")訪問了 

你能夠發現的是:以上的Tomcat6中的配置不論是全局仍是局部,都沒有修改項目的web.xml文件,可是仍然建議在web.xml中進行引用,主要是爲了項目的遷移,由於有的服務器須要在web.xml中進行聲明! 

總之:各類服務器有時候確實有比較大的出入,特別是感受J2EE中相似<resource-ref>這樣的一些標籤仍然不是很理解。html

相關文章
相關標籤/搜索