`

关于JNDI的简介

阅读更多

1.       什么是 JNDI ( Java Naming and Directory Interface )

JNDI 是用于访问不同命名和目录服务的统一 API 接口;

2.       JNDI 的架构和原理

      1.       架构:

         第一层: java 代码,是访问 JNDI 的代码;

         第二层: JNDI API :统一的命名和目录服务接口

         第三层: JNDI Naming Manager : JNDI 管理器

         第四层: JNDI SPI :用于构建 JNDI 实现的框架,能够动态的插入命名和目录服务提供商的产品;

         第五层:命名和目录服务提供商的产品;

         命名和目录服务提供商的产品 , 例如: DNS,LDAP,NIS,NDS……..; 因为这些产品都是不同的提供商提供的产品,所以他们的命名和目录服务的标准不一致,各个目录服务采用的访问协议也是不一样的,所以,要是直接访问他们的话,就要编写不同的 java 代码来访问他们;因此, JNDI SPI 就解决了这个问题,它能动态的插入这些命名和目录服务,能够将其协议专属的目录产品集成到系统中,使得我们能只需要调用一个统一的 javaAPI 就能够访问插入的产品了;

         2 .原理:

               JNDI 其实很好理解,他就是一个树状的结构,它的最顶是一个 initialContext 节点,然后它的下面就是绑定的一些对象或是一些 subContext ,用 JNDI 树就能够查找到树中每一个绑定上的节点上的对象的引用;

3. 怎样连接到 JNDI 上?

                  Environment environment = new Environment() ;// 创建一个环境对象

                   environment.setProviderurl(“t3://www.blogjava.net”) ;

                   environment.setSecurityPrincipal(“todd”) ;// 认证的用户名

                   environment.setSecurityCredentials(“841026”) ;// 密码

                   Context context = environment.getInitialContext() ;

         // 创建一个 subContext 并绑定一个对象:

                   Context subContext = context.createSubcontext(“subContext”) ;

                   subContext.rebind(“newObject”,object) ;// 绑定的对象必须是可序列化的

                   subContext.close() ;

                   context.close() ;

4. 如何通过 JNDI 查找 bind 的对象?

         Object object = context.lookup(“User”) ;

         // 把 user 窄化到原型

         User user = (User)javax.rmi.PortableRemoteObject.narrow(object, User.class) ;

         context.close() ;

 

 

 

一、JNDI是什么?

      JNDI--Java 命名和目录接口(Java Naming and Directory Interface),是一组在Java应用中访问命名和目录服务的API。

二、JNDI好处

      解耦:通过注册、查找JNDI服务,可以直接使用服务,而无需关心服务提供者,这样程序不至于与访问的资源耦合!
JNDI优点 写道
JNDI优点
  包含了大量的命名和目录服务,使用通用接口来访问不同种类的服务;   可以同时连接到多个命名或目录服务上;   建立起逻辑关联,允许把名称同Java对象或资源关联起来,而不必知道对象或资源的物理ID。   JNDI程序包:   javax.naming:命名操作;   javax.naming.directory:目录操作;   javax.naming.event:在命名目录服务器中请求事件通知;   javax.naming.ldap:提供LDAP支持;   javax.naming.spi:允许动态插入不同实现。   利用JNDI的命名与服务功能来满足企业级APIs对命名与服务的访问,诸如EJBs、JMS、JDBC 2.0以及IIOP上的RMI通过JNDI来使用CORBA的命名服务。

 

三、JNDI架构与原理

      相比较架构与原理,更关注与使用,故略!

四、JNDI使用

      在J2EE容器(如weblogic、websphere、jboss等)中使用:

    在weblogic环境下查找tuxedo 连接    

//在weblogic环境查找tuxedo连接
Context ctx = new InitialContext();
TuxedoConnectionFactory tuxedoFactory = (TuxedoConnectionFactory) ctx.lookup("tuxedo.services.TuxedoConnection");

   在web容器查找数据源

Context ctx=new InitialContext();
DataSource ds=(Datasource)ctx.lookup("java:comp/env/jdbc/mydatasource");

   重点关注Context ctx = new InitialContext(),上面的代码在容器下能够很好的工作,尤其是查找数据源的代码,无论是在tomcat,还是在weblogic、jboss,但如果脱离了容器,我们将得到异常“NoInitialContextException”,这是为什么呢?

   原因很简单,就是不存在相关的context,其本质是JNDI的服务提供者环境,即谁将提供此环境!

   为了能够使上面的代码工作,我们需要使用带参的构造子InitialContext(Hashtable),指定JNDI服务提供者环境信息,以weblogic环境为例:

Hashtable env = new Hashtable();
env.put(Context.INITIAL_CONTEXT_FACTORY, "weblogic.jndi.WLInitialContextFactory");
env.put(Context.PROVIDER_URL,"t3://localhost:7001");
InitialContext ctx = new InitialContext(env);

    但上面的代码不适合WebSphere、jboss等环境,原因在于所需环境信息不同,那么,Context ctx = new InitialContext()是如何做到在各个j2ee容器下有效地呢?原因在于这个默认的构造子是从System.properties读取

相关的环境信息的,由此不难猜出在各种J2EE容器下,如weblogic、jboss,它们都自己为自己设置了相应的信息!下面是一个关于System.properties设置JNDI服务环境的例子:

System.setProperty(Context.INITIAL_CONTEXT_FACTORY, "weblogic.jndi.WLInitialContextFactory");
System.setProperty(Context.PROVIDER_URL, "t3://localhost:7001");
InitialContext ctx = new InitialContext();

   较为详细点的信息可以参看这里!

五、借助Spring完成JNDI查找

    对于手头的weblogic,给出常用的两个查找例子,做个备忘吧!

查找tuxedo connnection:

<bean id="tuxedoConnFactory" class="org.springframework.jndi.JndiObjectFactoryBean">  
      <property name="jndiName">
            <value>tuxedo/services/TuxedoConnection</value>
      </property>
      <property name="resourceRef">
          <value>false</value>
     </property>
     <property name="jndiEnvironment">
       <props>
            <!-- The value of Context.PROVIDER_URL -->
            <prop key="java.naming.provider.url">t3://localhost:7001</prop>
            <prop key="java.naming.factory.initial">weblogic.jndi.WLInitialContextFactory</prop>
        </props>
      </property>
</bean>

 

    查找DataSource:

<bean id="tuxedoConnFactory" class="org.springframework.jndi.JndiObjectFactoryBean">  
      <property name="jndiName">
            <value>java:comp/env/jdbc/myDatasource</value>
      </property>
</bean>

   更多关于spring jndi配置参看这里,下面是其摘录:
Spring中对于JNDI的访问,提供了便捷的方法,在Spring的org.springframework.jndi包中包含了所有的类。其中提供了一下核心类:

(1)JndiTemplate:它是这个包的核心类,用来简化对JNDI的操作。它提供了对于lookup、bind方法。同时,允许JndiCallback接口的实现在其提供的JNDI上下文中执行任何操作。

使用JndiTemplate实现InitialContext环境变量的配置

<bean id="queueTarget" class="org.springframework.jndi.JndiObjectTargetSource">

<property name="jndiName">

<value>queue/testQueue</value>

</property>

<property name="jndiTemplate">

<ref local="jndiTemplate"/>

</property>

</bean>

<bean id="jndiTemplate" class="org.springframework.jndi.JndiTemplate">

<property name="environment">

<props>

<prop key="java.naming.factory.initial">org.jnp.interfaces.NamingContextFactory</prop>

<prop key="java.naming.provider.url">jnp://localhost:1099</prop>

<prop key="java.naming.factory.url.pkgs">org.jboss.namingrg.jnp.interfaces</prop>

</props>

</property>

</bean>

(2)JndiObjectTargetSource

使用JndiObjectTargetSource,例如

<bean id="queueTarget" class="org.springframework.jndi.JndiObjectTargetSource">

<property name="jndiName">

<value>queue/testQueue</value>

</property>

</bean>

(3)JndiObjectFactoryBean:

使用JndiObjectFactoryBean,例如

<bean id="dataSource" class="org.springframework.jndi.JndiObjectFactoryBean">

<property name="jndiName">

<value>java:/MySqlDS</value>

</property>

</bean>

总结(事例):

数据源的配置:

(1)配置可以访问到同一应用服务器的jndi数据源

<bean id="dataSource" class="org.springframework.jndi.JndiObjectFactoryBean">

<property name="jndiName">

<value>jdbc/cqccms</value>

</property>

</bean>

(2)配置能访问远程jndi数据源

<bean id="dataSource" class="org.springframework.jndi.JndiObjectFactoryBean">

<property name="jndiName">

<value>jdbc/cqccms</value>

</property>

<property name="jndiEnvironment">

<props>

<prop key="java.naming.factory.initial">

weblogic.jndi.WLInitialContextFactory

</prop>

<prop key="java.naming.provider.url">t3://172.16.101.42:7001</prop>

<prop key="java.naming.security.principal">weblogic</prop>

<prop key="java.naming.security.credentials">weblogic</prop>

</props>

</property>

</bean>

事务的配置:

不过事务对象远程访问好像没那么顺利,同理查看 org.springframework.transaction.jta.JtaTransactionManager类,没有发现上述方法,不过看到了setJndiTemplate()方法,也不错现配置一个jndiTemplate,如下:

<bean id="jndiTemplate" class="org.springframework.jndi.JndiTemplate" singleton="true"

lazy-init="default" autowire="default" dependency-check="default">

<property name="environment">

<props>

<prop key="java.naming.factory.initial">

weblogic.jndi.WLInitialContextFactory

</prop>

<prop key="java.naming.provider.url">t3://172.16.101.42:7001</prop>

<prop key="java.naming.security.principal">weblogic</prop>

<prop key="java.naming.security.credentials">weblogic</prop>

</props>

</property>

</bean>

然后在配置一下transactionManager,如下

<bean id="transactionManager" class="org.springframework.transaction.jta.JtaTransactionManager" singleton="true"

lazy-init="default" autowire="default" dependency-check="default">

<property name="jndiTemplate">

<ref local="jndiTemplate" />

</property>

<property name="userTransactionName">

<value>weblogic/transaction/UserTransaction</value>

</property>

</bean>

不过JtaTransactionManager默认找jndi name为UserTransaction的jta对象,在同一应用服务器上可以,远程访问不到,后直接查看weblogic的jndi树找到这个 jndi名weblogic/transaction/UserTransaction,配置上测试果然成功。

分享到:
评论

相关推荐

    JNDI简介与SPI实现

    **JNDI简介** Java Naming and Directory Interface (JNDI) 是Java平台提供的一种标准服务接口,用于在分布式环境中查找和管理资源。JNDI的主要目的是为了统一不同类型的命名和目录服务,使得开发者可以通过相同的...

    JNDI简介

    JNDI 简介 JNDI(Java Naming and Directory Interface)是一组用于从 Java 应用程序中访问名称和目录服务的一组 API。它提供了一个标准的、与命名系统无关的 API,这个 API 构建在特定于命名系统的驱动程序之上。...

    Jndi的使用简介

    简要描述项目中Jndi的使用过程,完成servlet通过池连接的方式访问数据库

    java JNDI总结

    #### 一、JNDI简介 **JNDI**(Java Naming and Directory Interface)是Java平台中的一个API,用于访问命名和目录服务,这些服务可以存储对象引用及其名称。JNDI的主要作用在于提供了一种在分布式环境中查找和访问...

    JNDI数据源配置

    #### 二、JNDI简介 JNDI是一种标准的API,用于访问命名和目录服务。它允许开发者使用统一的方式在分布式环境中查找和引用远程对象。对于数据库连接而言,这意味着可以将连接配置信息从应用程序代码中分离出来,存储...

    javaJNDI连接数据库

    #### JNDI简介 Java Naming and Directory Interface (JNDI) 是一种在Java环境中访问命名和目录服务的API。它为开发人员提供了查找和引用远程对象的标准方法,这包括数据库资源、消息队列和其他服务。JNDI的一个主要...

    目录服务和+JNDI

    - **集中管理**:集中存储关于网络对象的信息。 - **安全性**:提供认证和授权机制。 - **可扩展性**:能够轻松添加新的信息类型和服务。 #### 三、JNDI与LDAP的结合 本章节将详细介绍如何使用JNDI操作LDAP服务器...

    Jndi使用方法

    **1.3 LDAP简介** 轻量目录访问协议(LDAP)是目录服务的一种标准化协议,常用于访问和管理目录信息。JNDI通过LDAP接口实现了与LDAP服务器的通信,简化了开发者的任务。LDAP允许快速检索和更新目录数据,且能与其他...

    Jboss_jndi.zip

    1. **JNDI简介** JNDI是一个接口规范,它提供了查找和绑定Java对象的命名和目录服务。在Java EE应用中,JNDI常用来查找和管理数据源、EJB等资源。通过JNDI,我们可以将数据库连接信息抽象出来,使应用更具可移植性...

    Java演示JNDI的使用.rar

    1. JNDI简介: JNDI是Java平台的一部分,允许开发者查找和绑定数据,这些数据可以是任何类型,比如对象、配置信息、数据库连接等。JNDI通过提供统一的接口,使得开发者不必关心底层服务的具体实现,可以方便地接入...

    jndi配置

    #### 一、JNDI简介 Java Naming and Directory Interface (JNDI) 是一种为应用程序提供命名和目录服务的 Java API。它允许开发人员以一致的方式访问多种命名和目录服务,如 DNS、LDAP 和 RMI 等。在企业级应用开发...

    J2EE JNDI配置原理详解 JBOSS安装配置 Maven入门 Ant使用入门

    1.5 JSP 隐式对象简介 1.6 Java解析XML的四种方法 1.7 struts1和truts2比较 1.8 Tomcat5启动流程与配置详解 1.9 HttpServlet详解 1.10 Ant使用入门 1.11 Maven Ant之间的比较 1.12 详解MANIFEST.MF文件 1.13...

    jndi.zip_java ldap_jndi_jndi ldap_ldap_ldap java

    2. **LDAP简介** LDAP是一种基于X.500标准的轻量级目录访问协议,用于存储和检索分布式目录信息。它的设计目标是提供快速的读取操作,主要用于身份验证、授权和信息检索。在企业环境中,LDAP常用于存储用户账户、...

    Jndi数据连接池Jndi 数据连 接池

    #### 二、JNDI简介 JNDI是一种Java平台的标准接口,用于访问命名和目录服务。它提供了一种查找和获取命名对象的方式,这些对象可以是本地的也可以是远程的。在Web应用程序中,JNDI经常被用来查找数据库连接等资源。...

    SpringLDAP和JNDI的增删改查

    ### Spring LDAP简介 Spring LDAP是Spring框架的一个扩展,它为LDAP(Lightweight Directory Access Protocol)提供了一个面向对象的编程模型。Spring LDAP使得开发者可以使用熟悉的Spring编程风格来处理LDAP操作,...

    JNDI参考资料.doc

    **JNDI 简介** Java Naming and Directory Interface (JNDI) 是Java平台提供的一种标准接口,用于实现命名和目录服务。JNDI的主要目的是为了管理和访问各种类型的资源和服务,如数据库连接、电子邮件服务器、...

Global site tag (gtag.js) - Google Analytics