JSP 2.0的新特性

JSP 2.0属于J2EE 1.4平台,它在JSP 1.2基础之上增加了新的功能。它保证了向下兼容,原先使用的JSP技术在JSP 2.0中都可以支持。JSP 2.0的新功能主要包括下面几部分:

(一)运行环境变化

1、web.xml格式变化

我们知道JSP 1.2可以在Java 2标准版1.3版本运行,而JSP 2.0要求使用Java 2标准版1.4或更新版本,JSP 2.0使用由Servlet 2.4规定的Web程序部署描述格式。

在Web程序描述文件web.xml中需要使用xml schema打头的格式。在web.xml中主要的变化是所有有关JSP的设置信息要放在标记中。下面程序例1显示了一个web.xml大致的样子。

例1:

<?xml version="1.0" encoding="IS0-8859-1"?>
<web-app 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 web-app_2_4.xsd"
version="2.4">
.......
<jsp-config>
<taglib>
<taglib-uri>
http://www.icconcept.com/ics/sample-taglib
</taglib-uri>
<taglib-location>
/WEB-INF/jsp/sample-taglib.tld
</taglib-location>
</taglib>
........
<jsp-property-group>
<description>
For config the ICW sample application
</description>
<display-name>JSPConfiguration</display-name>
<url-pattern>/jsp/datareset.jsp</url-pattern>
<el-ignored>true</el-ignored>
<page-encoding>ISO-8859-1</page-encoding>
<scripting-invalid>true</scripting-invalid>
<include-prelude>/jsp/prelude.jspf</include-prelude>
<include-coda>/jsp/coda.jspf</include-coda>
</jsp-property-group>
</jsp-config>
</web-app>

2、JSP设置

在标记提供了Web程序中JSP程序的设置信息。包括和两类元素。定义了Web程序使用的custom tag,它的用法和以前JSP 1.2中的用法相同。定义了一组JSP的特性。这些特性实际上对应JSP的page directive定义的特性。通过只不过可以方便地对多个具有相同属性的JSP统一定义。

定义一个或多个URL样式,在中定义的属性会适用于匹配这些URL样式的所有JSP文件。在中的属性可以定义以下设置:

(1)允许或禁止使用表达式语言(EL)

在中可以设定是否允许对应的JSP使用JSTL表达式语言(EL)。如果属性标记被设定为false,JSP中的EL表达式会被处理;如果是true,Web容器会在转换JSP时忽略EL表达式。

(2)允许或禁止使用scripting

属性可以允许或禁止使用JSP的脚本语言(scripting)。如果这个属性标记对应为true,即scripting元素被禁止,则JSP中不能使用scriptlet,scripting表达式和declaration,否则会有转换错误。当这个属性标记为false时,JSP可以像在1.2版本之前那样使用脚本语言。

(3)声明JSP编码

通过标记可以设置对应的JSP网页的编码。这个属性对应每个JSP中的pageEncoding属性,Web容器将根据这个属性对JSP内容进行编码。

(4)对应隐含包括(Implicit Includes)

在中可以在对应JSP中加入抬头(preludes)和结尾(coda),使用和属性可以设定在JSP网页中包括的preludes和coda的jspf文件。这些文件的位置相对于当前Web程序的context。当有超过一个preludes或coda元素在中时,JSP会按照其顺序加入到内容中。

(二)引入表达式语言(EL)

JSP 2.0的一个主要特点是它支持表达语言(expression language)。JSTL表达式语言可以使用标记格式方便地访问JSP的隐含对象和JavaBeans组件,JSTL的核心标记提供了流程和循环控制功能。自制标记也有自定义函数的功能,因此基本上所有seriptlet能实现的功能都可以由JSP替代。在JSP 2.0中,建议尽量使用EL而使JSP的格式更一致。

在web.xml的中可以控制一组JSP是否使用EL,在每个JSP中也可以指定是否该JSP使用EL。在page directive中的isELIgnored属性用来指定是否忽略。格式为:

<%@ page isELIgnored="true|false"%>

如果设定为真,那么JSP中的表达式被当成字符串处理。比如下面这个表达式
${2000 % 20}

在isELIgnored="true"时输出为${2000 % 20},而isELIgnored="false"时输出为100。Web容器默认isELIgnored="false"。

虽然JSP 2.0可以使JSP中完全使用表达语言而避免scriptlet,在实际编程中,应该根据程序的功能要求和编程人员的自身条件选择合适的方式。使用表达语言的JSP比较方便规整,但是由于需要将标记进行转换,在第一次被调用时会比较慢;有些编程人员由于对Java比较了解,因而更习惯JSP 1.2之前的编程方式,因此,在使用中应因地制宜地选择适用的编程方法。

(三)SimpleTag

JSP 2.0中加入了新的创建自制标记的API,javax.servlet.jsp.tagext.SimpleTag定义了用来实现简单标记的接口。和JSP 1.2中的已有接口不同的是,SimpleTag接口不使用doStartTag()和doEndTag()方法,而提供了一个简单的doTag()方法。这个方法在调用该标记时只被使用一次。而需要在一个自制标记中实现的所有逻辑过程、循环和对标记体的评估等都在这个方法中实现。从这个方面来讲,SimpleTag和IterationTag可以达到同等的作用。但SimpleTag的方法和处理周期要简单得多。在SimpleTag中还有用来设置JSP内容的seUspBody()和getJspBody()方法。Web容器会使用setJspBody()方法定义一个代表JSP内容的JspFragment对象。实现SimpleTag标记的程序可以在doTag方法中根据需要多次调用getJspBody().invoke()方法以处理JSP内容。

例如,程序例2 SimpleTag根据指定的次数(times)进行循环并输出当前序号(sequence)。程序的结构比较简单,所有逻辑都在doTag方法中实现。

例2:

package ICW.taglib;
import javax.servlet.jsp.JspException;
import javax.servlet.jsp.tagext.SimpleTagSupport;
import java.util.HashMap;
import java.io.IOException;
public class IterationSimpleTag extends SimpleTagSupport{
private int times;
public void setTimes(int_times){
this.times=_times;
}
public void doTag() throws JspException,IOException{
HashMap params=new HashMap();
for(int i=0; i<times;i++){
params.put("sequence",String.valueOf(i+1));
getJspBody().invoke(null,params);
}
}
}

这个标记的TLD文件内容如下,它使用了XML schcma定义标记的使用方法。

程序例3如下:

<?xml version="1.0" encoding="UTF-8" ?>
<taglib xmlns="http://java.sun.com/xml/ns/i2ee"
xmlns:xsi="http://WWW.w3.org/2001/XMLSchema-instance"
xsl:schemaLocation="http://java.sun.com/xml/ns/j2ee web-jsptaglihrary_2_0.xsd"
version="2.0">
<taglib>
<tiib-version>1.0</tlib-version>
<short-name>Jwad book simple tag</short-name>
<uri>/JwadSimpleTag</uri>
<description>Simple Tag Handler</description>
<tag>
<name>iteration</name>
<tag-class>ICW.taglib.IterationSimpleTag</tag-class>
<body-content>scriptless</body-content>
<description>Iteration Tag</description>
<variable>
<description>Current iterationnumber</description>
<name-given>sequence</name―given>
</variable>
<attribute>
<name>times</name>
<required>true</required>
<rtexprvalue>true</rtexprvalue>
</attribute>
</tag>
</taglib>

程序例4中的JSP使用上面例3中定义的IterationSimpleTag,它根据Web请求参数中给定的“times”的值进行一定次数的循环。在每次循环中将输出"sequence"的值。

例4:

<%@ taglib prefix="ictag" uri="/WEB-INF/ics-jsp2.tld" %>
<HTML><HEAD><TITLE>Simple Tag Sample</TITLE></HEAD>
<BODY>
<CENTER>
<FONT COLOR='#009999' SIZE='4' face='Arial'>
<STRONG>Interation Simple Tag</STRONG>
</FONT>
</CENTER>
<HR>
<c:set var="time" value="${param.times}"/>
<p><B>Reminder:</B></p><br>
<ictag:iteration times="${times}">
This is the ${sequence} Of ${times} times of reminder<br>
</ictag:iteration>
</body>
</html>

(四)使用JSP fragment

JSP 2.0中的一个主要功能是JSP fragment,它的基本特点是可以使处理JSP的容器推迟评估JSP标记属性。我们知道一般JSP是首先评估JSP标记的属性,然后在处理JSP标记时使用这些属性,而JSP fragment提供了动态的属性。也就是说,这些属性在JSP处理其标记体时是可以被改变的。JSP需要将这样的属性定义为javax.servlet.jsp.tagext.JspFragment类型。当JSP标记设置成这种形式时,这种标记属性实际上的处理方法类似于标记体。在实现标记的程序中,标记属性可以被反复评估多次。这种用法称为JSP fragment。JSP fragment还可以定义在一个SimpleTag处理程序中使用的自制标记动作。像前面例子说明的,getJspBody返回一个JspFragment对象并可以在doTag方法中多次使用。需要注意的是,使用JSP fragment的JSP只能有一般的文本和JSP action,不能有scriptlet和scriptlet表达式。

我们可以简单地认为JSP fragment是一段可以重复使用的JSP。一段JSP fragment可以被传递到另一个JSP中并被使用。与JSP include概念不同的是,JSP fragment一般功能比较短小简单而且重复使用率高。

JSP fragment一般在标记体内或标记体内定义。每次当含有JSP fragment的标记被使用时,Web容器生成一个JspFragment对象,这个对象和当前JSP的page scope联系起来。同时,这个JspFragment对象和包含它的父标记建立联系。JspFragment对象可以有两种调用方式:使用Java编写的标记处理程序,或者使用标记文件(tag file)。标记文件可以使用,或者动作使用JSP fragment。 JSP标记文件可以和JSP fragment共同工作。CustomTag都是以编程方式实现的。JSP标记文件是用文本文件格式(JSP语法)实现自制标记,这也是JSP 2.0的一个主要新功能。一个标记文件可以对应一个标记,可以不需tld文件定义该标记的方法。这样,即使编程人员不熟悉Java,也可以使用JSP语法定义自己的标记。标记文件一般使用.tag为后缀并放在Web程序的/WEB-INF目录下。

程序例5中的taskstatus.jsp使用了两个JSP fragment。这个JSP的功能是显示一组Task的名称和完成日期,它通过定义了两段JSPfragment(名称为onSehedule和delayed)。在标记内的JSP就是JSPfragment,而标记被一个包围。这个标记是一个通过标记文件定义的自制标记,它的定义文件在/WEB-INF/tags目录下。标记文件的名称和标记名称一致为“listTasks.tag"。这个标记会使用到前面定义的两个JSP fragment。

例5:

<%@ taglib prefix="ictag" tagdir="/WEB-INF/tags" %>
<HTML><HEAD><TITLE>JSP Fragment Sample</TITLE></HEAD>
<BODY>
<CENTER>
<FONT COLOR='#009999' SIZE='4' face='Arial'>
<STRONG>JSP Fragment Sample Using Tag Files</STRONG>
</FONT>
</CENTER>
<HR>
<h2>Tasks</h2>
<ietag:listTasks>
<jsp:attribute name="onSchedule">
<td>
Name:${name}<br/>
</td><td>
Date:${date}
</td>
</jsp:attribute>
<jsp:attribute name="delayed">
<td>
Name:${name}<br/>
</td><td>
<font color="red">Plan:<strike> ${pianDate}</strike></font><br/>
<b>Actural:${actDate}</b>
</td>
</jsp:attribute>
</ictag:listTasks>
</BODY>
</HTML>

(五)其他特性

JSP2.0还有一些其他特性变化,比如严格修正了I18N的语法规则,改进JSP对应XML语法从而允许使用namespaces等。这些并不是核心功能,大家可以参照java.sun.com的有关资料了解功能的细节,这里就不再阐述。


skygragon @ 2005-10-11 10:31

1、XML Schema定义Web查询部署描述文件

Servlet 2.3之前的版本使用DTD作为部署描述文件的定义,其web.xml的格式为如下所示:

<?xml version="1.0" encoding="IS0-8859-1"?>
<!DOCTYPE web-app
PUBLIC "-//sunMicrosystems,Inc.//DTD WebApplication 2.3f//EN"
"http://java.sun.com/j2ee/dtds/web-app_2.3.dtd">
<web-app>
.......
</web-app>



Servlet 2.4版首次使用XML Schema定义作为部署描述文件,这样Web容器更容易校验web.xml语法。同时XML Schema提供了更好的扩充性,其web.xml中的格式如下所示:

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:workflow="http://www.workflow.com"
xmins: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">
.........
</web-app>



虽然上面显示两个版本的开头几行有差别,Servlet 2.4版程序web.xml中Web构件的设置方法和Servlet 2.3版大致相同。

2、Servlet Request监听器

Servlet 2.4版在事件监听器中加入了ServletRequest监听器,包括:ServletRequestListener,ServletRequestAttributeListener和其他相关类。这些类可以用来管理和控制与ServletRequest动作有关的事件。下面的程序显示了一个典型的ServletRequest监听器的结构。

import javax.setvlet.ServletContext;
import javax.servlet.ServletRequestListener;
import javax.servlet.ServletRequestAttributeListener;
import javax.servlet.http.HttpServletRequest;
import iava.io.*;
import java.util.Locale;
public final class RequestListener implements SerVletRequestListener,
ServletRequestAttributeListener,ServletContextListener{
........
public void requestlnitialized(javax.servlet.ServletRequestEvent event){
........
}
public void attributeAdded(javax.servlet.ServletRequestAttributeEvent event){
........
}
public void attributeRemoved(javax.servlet.ServletRequestAttributeEvent event){
........
}
public void attributeReplaced(javax.servlet.ServletRequestAttributeEvent event){
........
}
public void attributeDestroyed(javax.servlet.ServletRequestAttributeEvent event){
........
}
}



3、 Request Dispatcher变化

Servlet 2.4版的Web程序增强了filter和request dispatcher的配合功能,这样过滤器可以根据请求分发器(request dispatcher)所使用的方法有条件地对Web请求进行过滤。编程者可以通过在web.xml中使用元素设定过滤器作用的条件(下图1):




图1 设定元素的过滤器


·只有当request直接来自客户,过滤器才生效,对应为REQUEST条件。

·只有当request被一个请求分发器使用forward()方法转到一个Web构件时(采用或定义),对应称为FORWARD条件。

·类似地,只有当request被一个请求分发器使用include()方法转到一个Web构件时(采用或定义),对应称为INCLUDE条件。

·只有当request被一个请求分发器使用“错误信息页”机制方法转到一个Web构件时,对应称为ERROR条件。

·第五种过滤器作用的条件可以是上面四种条件的组合。

下面的程序中的定义了当客户请求/icsamples/* 样式的URL时,Security Filter会被用来过滤请求。但是如果到达URL为/icsamples/* 的Web构件的请求是从一个request dispatcher转发过来的,这个过滤器不工作。

<filter-mapping>
<filter-name>Security Filter</filter-name>
<url-pattern>/icsamples/*</url-pattern>
</filter-mapping>



如果使用下面的程序设置,在中加入INCLUDE,Security Filter只有在被一个request dispatcher使用include()方法包括RequestRecorderServlet时才会工作。在其他情况(如请求直接从客户发来,或request dispatcher使用forward方法)下,Security Filter都不会工作。

<filter-mapping>
<fliter-name>Security Filter</filter-name>
<servlet-name>RequestRecorderServlet</servlet-name>
<dispatcher>INCLUDE</dispatcher>
</filter-mapping>



下面的程序设置定义了Security Filter当request由客户直接发出或request

dispatcher使用forward方法时能工作。

<filter-mapping>
<filter-name>Security Filter</filter-name>
<url-pattern>/icsamples/*</uri-pattern>
<dispatcher>FORWARD</dispatcher>
<dispatcher>REQUEST</dispatcher>
</filter-mapping>



程序例6显示了一个在中使用的例子。程序7定义的ReqDispatcherServlet根据用户请求的参数"type",决定如何处理请求。在其中使用了两个RequestDispatcher(rd,和rd2),它们可以向URL为/admin或/control的servlet转发请求。当请求被转发后,适当的过滤器会起作用。过滤器和Web资源的关系可以在web.xml中定义。如果用户输入的Web请求参数为“INCLUDE”,ReqDispatcherServlet的RequestDispatcher会调用include 方法,这样DispatcherFilterIcd会被使用,因为在定义了INCLUDE;如果用户输入的 Web请求参数为“ERROR”ReqDispatcherServlet会调用resp.sendError()方法,这样DispatcherFilterErr会被使用,因为在定义了ERROR。

例6:

<filter-mapping>
<filter-name>DispatcherFilterIcd</filter-name>
<url-pattern>/admin</url-pattern>
<dispatcher>INCLUDE</dispatcher>
</filter-mapping>
<filter-mapping>
<filter-name>DispatcherFilterErr</filter-name>
<url-pattern>/errorpage</url-pattern>
<dispatcher>ERROR</dispatcher>
</filter-mapping>



例7:

import javax.servlet.*;
import javax.servlet.http.*;
import java.io.*;
import java.util.*;

public class ReqDispatcherServlet extends HttpServlet
{
String dispatchtype;
public void init(ServletConfig config) throws ServletException
{
super.init(config);
}
public void service(HttpServletRequest req,HttpServletResponse resp)
throws ServletException,IOException
{
PrintWriter out=resp.getWriter();
String type="NONE";
if((req.getParameter("type"))!=null)
{
type = req.getParameter("type");
}
resp.setContentType("text/html");
out.println("<HTML>");
out.println("<BODY>");
out.println("<HR>");
out.println("<p>");
out.println("ReqDispacherServelt");
out.println("</P><p>");
out.println("ServerName:"+req.getServerName()+"ServerPort:"+
req.getServerPort());
out.println("</p>");
RequestDispatcher rd=req.getRequestDispatcher("/admin");
RequestDispatcher rd2=req.getRequestDispatcher("/control");
if(type.equals("REQUEST"))
{
}
if(type.equals("FORWARD"))
{
rd.forward(reg,resp);
}
if(type.equals("INCLUDE"))
{
rd.include(req,resp);
}
if(type.equals("ERROR"))
{
resp.sendError(404,"Error from ReqDispacherServlet");
}
if(type.equals("CONTROL"))
{
rd2.forward(req,resp);
}
out.flush();
}
public voiddestroy()
{
System.out.println("ReqDispacherServlet:destroy()");
}
}



4、 增强的国际化功能

Servlet 2.4增加了Web程序国际化功能,在web.xml中可以定义网站的字符编码方式。

<locale-encoding-mapping-list>
<locale-encoding-mapping>
<locale>zh</locale>
<encoding>gb2312</encoding>
</locale-encoding-mapping>
</locale-encoding-mapping-list>



当客户请求了特定语言的Web资源时,servlet程序通过ServletResponse接口的setLocale方法设置一个Web响应的语言属性。

5、 Login/Logout功能

在Servlet 2.4中增加了logout和login方法方便安全管理。大家可以参考Servlet 2.4的API查看其用法。

你可能感兴趣的:(编程,jsp,xml,Web,servlet)