很多地方都会用到JNDI,一大堆的缩写加上一大堆不清不楚的概念描述,使得在看到的时候都不认识,更不要说使用了。
 
JNDI,Java Naming Directory Interface,J2EE的标准之一,所有的J2EE容器都必须提供一个JNDI的服务,但是,我一直都没用过,至少是没有刻意地去用过。因为,我也 曾经把数据源配置在Tomcat的JNDI服务中,但那时,我也只是刚刚涉足JAVA,有人告诉我应该这么干而已。后来,我开始使用自定义的数据库连接配 置文件,就再也没有跟JNDI打过交道了,直到今天。


现在之所以又想看一下,只是因为觉得这是J2EE的重要标准之一,如果不懂得的话,似乎有点说不过去。


JNDI
的主要功能可以这样描述,它使用一张哈希表存储对象(大多数的J2EE容器也的确是这样做的),然后,开发人员可以使用键值——也就是一个字符串——来获
取这个对象。这里就包括取JNDI的两个最主要操作,bind和lookup。bind操作负责往哈希表里存对象,存对象的时候要定义好对象的键值字符
串,lookup则根据这个键值字符串往外取对象。


JNDI的命称可能会让人产生混淆,似乎觉得这是一个用来操作目录的,事实上,我更愿
意把这个目录理解成为JNDI存放对象时使用的格式,也就是说,JNDI以目录的方式存储对象的属性。例如,用户通过JNDI存储一个汽车对象,那么,汽
车就是根目录,汽车的轮子、引擎之类的子对象就算是子目录,而属性,比如说汽车的牌子、重量之类,就算是汽车目录下的文件。


JNDI的功能既然就是根据一个字符串键值就可以取得一个想要得到的对象,我一开始就觉得这不是跟COM或CORBA一样吗?SUN也是有野心的企业啊,JNDI应该就是它要努力推行的JAVA下的分布式开发的标准吧。


JNDI
的出现应该就是为了分步式开发服务的,有人负责开发这种分布式对象,有人只需要使用这些分布式对象就可以了,这两组人不必属于同一个公司,而且这种开发通
常应该是不并行的,也不必是会了同一个项目服务。就如果数据源对象,它放在JNDI中,只要想要用的人,直接通过JNDI服务取来用就可以了,至于当初是
谁把它放进JNDI中的,还是不用操这份心了吧。而我一直没有使用JNDI,也就是这个原因,项目中的所有对象都在我控制之下,我不去使用别人的对象,也
没打算把我的对象贡献出来给别人使用,那自然也就没必要去跟JNDI打交道。我觉得是否使用JNDI,这应该是关键原因,至于什么方便性、安全性之类的考
虑,应该不是JNDI的主要目的,就如同你可以用JAVA来做网站,但JAVA并不是专门用来做网站的。


可能有人觉得这种功能跟IoC也
很象,这个我倒不觉得,虽然对于对象的使用人员来说的确是这种感觉,且不说IoC需要为对象定义接口,而JNDI并无此限制,先说这里有一个使用环境问
题,我觉得IoC是用来解决并行开发问题的,也就是说IoC主要是用于明确设计人员与实现/使用人员的分工,无论是设计的,还是使用的,通常是一个项目组
里的人,使用IoC,可以使得设计人员专注于设计,加快设计速度。因此,IoC的用途要比JNDI广泛的多,现在大型系统中,不使用IoC的,几稀矣。

JNDI 笔记(二) J2EE下使用JNDI

在J2EE环境下使用JNDI是非常简单的事,因为所有的J2EE容器都要实现JNDI服务,所以,在J2EE环境下使用JNDI,与使用
Hashtable也没有什么太大区别。只有一点限制,那就是绑定对象时,对象所属的类必须实现java.io.Serializable接口,这一点也
实在一点也不困难,几乎所有用到的Java类都实现了这个接口,对于自定义的类,在接口实现列表里把这个接口加进去也就是了。

下面,我将演示一下如何在J2EE环境下使用JNDI,为了保证代码的通用性,我不使用struts之类的框架,而是直接使用标准JSP和Servlet实现。我将该项目的名称定为jndi_test

要使用JNDI,需要先到SUN的网站上去下载jndi.jar。

 2.1 JSP

本项目包括5个JSP,功能说明如下:

  • index.jsp:首页
  • bind.jsp:用于在JNDI中绑定对象
  • bind_result.jsp:绑定对象后的返回页面
  • lookup.jsp:用于在JNDI中检索对象
  • lookup_result.jsp:用于显示检索对象

本节中用到的JSP代码如下,代码都简单地很,就不多做解释了。

2.1.1 index.jsp


>
bind an object
</
a
>

>
lookup the binded object
</
a
>

</
body
>

</
html
>

2.1.5 lookup_result.jsp

<%

@ page language
=
"
java
"
 contentType
=
"
text/html; charset=GB18030
"


    pageEncoding
=
"
GB18030
"

%>

<!
DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"
>

<
html
>

<
head
>

<
meta 
http-equiv
="Content-Type"
 content
="text/html; charset=GB18030"
>

<
title
>
JNDI Test - Lookup result
</
title
>

</
head
>

<
body
>

<%


    
Object
 o 
=
 request.getAttribute(
"
found_jndi_obj
"
);

    out.println(o);

%>

</
body
>

</
html
>

2.2 Servlet

本例包括两个Servlet,功能说明如下:

  • BindServlet:用于在JNDI服务中绑定一个对象
  • LookupServlet:用于在JNDI服务中取出一个对象

2.2.1 BindServlet.java


package
 lld.test.jndi;


import
 java.io.IOException;

import
 java.util.Date;


import
 javax.naming.Context;

import
 javax.naming.InitialContext;

import
 javax.servlet.RequestDispatcher;

import
 javax.servlet.ServletContext;

import
 javax.servlet.ServletException;

import
 javax.servlet.http.
*
;


public
 
class
 BindServlet 
extends
 HttpServlet

{



    
private
 
static
 
final
 
long
 serialVersionUID 
=
 
5219969790998794367L
;



    @Override

    
protected
 
void
 doGet(HttpServletRequest req, HttpServletResponse resp)

            
throws
 ServletException, IOException

{

        
this
.doPost(req, resp);

    }




    @Override

    
protected
 
void
 doPost(HttpServletRequest req, HttpServletResponse resp)

            
throws
 ServletException, IOException

{

        
try

{

            Context jndi_ctx 
=
 
new
 InitialContext();

            String key 
=
 
"
jndi_object
"
;

            jndi_ctx.rebind(key, 
new
 Date());

        }

catch
(Exception ex)

{

            ex.printStackTrace();

        }


        

        ServletContext context 
=
 
this
.getServletContext();

        RequestDispatcher dispatcher 
=
 context.getRequestDispatcher(
"
/bind_result.jsp
"
);

        dispatcher.forward(req, resp);

    }


    

}

使用rebind而不是bind绑定对象是因为,使用bind时,如果已经有对象绑定到该键值上,则会抛出异常。

因为只是示例代码,所以我只是绑定了一个最简单的日期对象。

2.2.2 LookupServlet.java


package
 lld.test.jndi;


import
 java.io.IOException;


import
 javax.naming.Context;

import
 javax.naming.InitialContext;

import
 javax.servlet.RequestDispatcher;

import
 javax.servlet.ServletContext;

import
 javax.servlet.ServletException;

import
 javax.servlet.http.HttpServlet;

import
 javax.servlet.http.HttpServletRequest;

import
 javax.servlet.http.HttpServletResponse;


public
 
class
 LookupServlet 
extends
 HttpServlet

{

    
private
 
static
 
final
 
long
 serialVersionUID 
=
 
6677219828267184673L
;



    @Override

    
protected
 
void
 doGet(HttpServletRequest req, HttpServletResponse resp)

            
throws
 ServletException, IOException

{

        
this
.doPost(req, resp);

    }




    @Override

    
protected
 
void
 doPost(HttpServletRequest req, HttpServletResponse resp)

            
throws
 ServletException, IOException

{

        
try

{

            Context jndi_ctx 
=
 
new
 InitialContext();

            String key 
=
 
"
jndi_object
"
;

            Object o 
=
 jndi_ctx.lookup(key);

            req.setAttribute(
"
found_jndi_obj
"
, o);

        }

catch
(Exception ex)

{

            ex.printStackTrace();

        }


        

        ServletContext context 
=
 
this
.getServletContext();

        RequestDispatcher dispatcher 
=
 context.getRequestDispatcher(
"
/lookup_result.jsp
"
);

        dispatcher.forward(req, resp);

    }


    

}

2.3 web.xml

在web.xml中,加入了servlet映射


<?
xml version="1.0" encoding="UTF-8"
?>

<
web-app 
id
="WebApp_ID"
 version
="2.4"
 xmlns
="http://java.sun.com/xml/ns/j2ee"
 xmlns:xsi
="http://www.w3.org/2001/XMLSchema-instance"
 xsi:schemaLocation
="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"
>


    
<
display-name
>
jndi_test
</
display-name
>


    

    
<
servlet
>


        
<
servlet-name
>
BindServlet
</
servlet-name
>


        
<
servlet-class
>
lld.test.jndi.BindServlet
</
servlet-class
>


    
</
servlet
>


    
<
servlet-mapping
>


        
<
servlet-name
>
BindServlet
</
servlet-name
>


        
<
url-pattern
>
/bind.do
</
url-pattern
>


    
</
servlet-mapping
>


    

    
<
servlet
>


        
<
servlet-name
>
LookupServlet
</
servlet-name
>


        
<
servlet-class
>
lld.test.jndi.LookupServlet
</
servlet-class
>


    
</
servlet
>


    

    
<
servlet-mapping
>


        
<
servlet-name
>
LookupServlet
</
servlet-name
>


        
<
url-pattern
>
/lookup.do
</
url-pattern
>


    
</
servlet-mapping
>


    

    
<
welcome-file-list
>


        
<
welcome-file
>
index.jsp
</
welcome-file
>


    
</
welcome-file-list
>

</
web-app
>


OK,所有的代码都在这里了,部署到Tomcat下运行即可。

JNDI 笔记(三) J2SE下使用JNDI

在J2SE下使用JNDI下就显得困难一些,首先,我们没有单独的JNDI服务器可以用,JBoss提供了一个免费的JNP服务,通过配置可以作为
单独的JNDI服务器启用。不过这里就不这么麻烦了,如何使用JBOSS作为JNDI服务器,以后将单独撰文讲述,这里我使用sun提供的
com.sun.jndi.fscontext.RefFSContextFactory作为JNDI服务器,其实这是使用文件系统来存储JNDI对象。
至于如何存储后文还将专门描述。

为了在J2SE下使用JNDI,我们首先得到sun的网站上下载3个包,jndi.jar、fscontext.jar和providerutil.jar,前者提供了JNDI服务的接口,后两者是我们要使用的文件系统作为JNDI服务器的支持包。

使用RefFSContextFactory,要求绑定的对象必须实现javax.naming.Referencable接口,否则在绑定时将报如下错误:

Can only bind References or Referenceable objects

各个JDBC驱动提供商提供的DataSource类都实现了Referencable接口,可以直接使用。不过本着学习的态度,我还是在这里演示一下如何实现Referencable接口。

这个如何实现将在后文结合代码详细介绍。本例包括4个类,说明如下:

  • BindedClass:自定义的实现Referenceable接口的类
  • BindedClassFactory:工厂类,能够把一个Reference对象转换为BindedClass对象
  • Bind:测试类,用于在JNDI中绑定对象
  • Loopup:测试类,用于从JNDI中获取对象

3.1 BindedClass和BindedClassFactory

3.1.1 BindedClass


package
 lld.test.jndi;


import
 javax.naming.NamingException;

import
 javax.naming.Reference;

import
 javax.naming.Referenceable;

import
 javax.naming.StringRefAddr;


public
 
class
 BindedClass 
implements
 Referenceable 

{

    
public
 String value; 

    

    
public
 BindedClass()

{

    }


    

    @Override

    
public
 Reference getReference() 
throws
 NamingException

{

        Reference r 
=
 
new
 Reference(
this
.getClass().getName(), BindedClassFactory.
class
.getName(), 
null
);

        r.add(
new
 StringRefAddr(
"
value
"

this
.getValue()));

        
return
 r;

    }




    
public
 String getValue()

{

        
return
 value;

    }




    
public
 
void
 setValue(String value)

{

        
this
.value 
=
 value;

    }




}

3.1.2 BindedClassFactory


package
 lld.test.jndi;


import
 java.util.Hashtable;


import
 javax.naming.
*
;

import
 javax.naming.spi.
*
;


public
 
class
 BindedClassFactory 
implements
 ObjectFactory

{

    @Override

    
public
 Object getObjectInstance(Object obj, Name name, Context nameCtx,

JNDI 笔记(一) 概述的更多相关文章

  1. JNDI 笔记

    原理:         在DataSource中事先建立多个数据库连接,保存在数据库连接池中.当程序访问数据库时,只用从连接池中取空闲状态的数据库连接即可,访问结束,销毁资源,数据库连接重新回到连接池 ...

  2. ng-深度学习-课程笔记-0: 概述

    课程概述 这是一个专项课程(Specialization),包含5个独立的课程,学习这门课程后做了相关的笔记记录. (1) 神经网络和深度学习 (2)  改善深层神经网络:超参数调试,正则化,优化 ( ...

  3. Spring4笔记1--Spring概述、IoC

    Spring概述: Spring框架: Spring 由 20 多个模块组成,它们可以分为数据访问/集成(Data Access/Integration).Web.面向切面编程(AOP,  Aspec ...

  4. μCos-ii学习笔记1_概述

    一.μCos-ii _概述 网上关于μCosii的文章多不胜数,本人学习的过程中也参考了很多人的理解和想法,看的是卢有亮老师的<嵌入式实时操作系统-μC/OS原理与实践>(第2版),同时也 ...

  5. HTTP权威指南笔记-1.概述

    1.1 通讯 Web内容是存储在服务器上的,Web服务所使用的是HTTP协议,所以经常称为HTTP服务器.通讯过程为客户端(正常我们所使用的)发出请求,服务端根据客户端的HTTP请求响应相应数据,这就 ...

  6. XML学习笔记1——概述

    我对于XML是很不够重视的,认识也是非常肤浅的,因为在之前的Web经验中,基本上都可以使用JSON来代替XML,JSON网络流量少,解析快,JS支持好等这些特点让我对自己的观点坚信不疑.然而我渐渐地改 ...

  7. MyBatis学习笔记(一) 概述

    一.什么是MyBatis? MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBat ...

  8. Python学习笔记 - day1 - 概述及安装

    Python概述 Python是一种计算机程序设计语言.我们平时已经听说过很多种流行的编程语言,比如非常难学的C语言,非常流行的Java语言,适合网页编程的JavaScript语言等等. Python ...

  9. JSP笔记02——概述(转)

    不完全翻译,结合谷歌,一定主观性,还可能有误,原始内容地址:https://www.tutorialspoint.com/jsp/jsp_overview.htm 主要内容如下: 什么是JSP? 为什 ...

随机推荐

  1. 解决sql server死锁

    -- 查询死锁 select request_session_id spid,OBJECT_NAME(resource_associated_entity_id) tableName from sys ...

  2. jQuery——jQuery选择器

    基本选择器 # Id选择器 $(“#btnShow”)选择id为btnShow的一个元素 . 类选择器 $(“.liItem”)选择含有类liItem的所有元素 ele 标签选择器 $(“li”)选择 ...

  3. oracle 用户的操作

    语法: CREATE USER user   IDENTIFIED { BY password              | EXTERNALLY [ AS 'certificate_DN' ]    ...

  4. git clone下载代码,中途断掉怎么办?

    问题如下: 解决办法: 1)运行以下命令进行clone $ git clone --recursive https:xxxxxx 2)进入项目根目录,继续下载 $ cd eigen-git-mirro ...

  5. vue移动端地址三级联动组件(一)

    vue移动端地区三级联动 省,市,县.用的vue+mintUi 因为多级联动以及地区的规则比较多.正好有时间自己写了一个.有问题以及建议欢迎指出.涉及到dom移动,所以依赖vue+jquery.这边数 ...

  6. 【原创】使用HTML5+canvas+JavaScript开发的原生中国象棋游戏及源码分享

    目前已经实现的功能: V1.0 : 实现棋子的布局,画布及游戏场景的初始化V2.0 : 实现棋子的颜色改变V3.0 :实现所有象棋的走棋规则V4.0 : 实现所有棋子的吃子功能 GItHub源码下载地 ...

  7. MySQL之索引以及正确使用索引

    一.MySQL中常见索引类型 普通索引:仅加速查询 主键索引:加速查询.列值唯一.表中只有一个(不可有null) 唯一索引:加速查询.列值唯一(可以有null) 组合索引:多列值组成一个索引,专门用于 ...

  8. Python os.listdir() 方法

    概述 os.listdir() 方法用于返回指定的文件夹包含的文件或文件夹的名字的列表.这个列表以字母顺序. 它不包括 '.' 和'..' 即使它在文件夹中. 只支持在 Unix, Windows 下 ...

  9. Spring Boot学习总结(4)——使用Springloaded进行热部署

    我在开发的时候,总是会及时对自己的程序进行测试,总是频繁的重启web server,容器不烦我们都觉得烦了. dependencys目录下增加: <dependency> <grou ...

  10. 09springMVC对ajax的支持

    u  最直接的Ajax处理 u  数据绑定@RequestBody/@ResponseBody u  使用@RequestBody/@ResponseBody来支持Ajax u  使用HttpEnti ...