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。
本項目包括5個JSP,功能說明如下:
本節中用到的JSP代碼如下,代碼都簡單地很,就不多做解釋了。
</
body
>
</
html
>
<%
@ 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
>
本例包括兩個Servlet,功能說明如下:
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時,如果已經有對象綁定到該鍵值上,則會拋出異常。
因為只是示例代碼,所以我只是綁定了一個最簡單的日期對象。
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);
}
}
<?
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
>
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個類,說明如下:
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;
}
}
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,