|
![]() |
名片设计 CorelDRAW Illustrator AuotoCAD Painter 其他软件 Photoshop Fireworks Flash |
|
即将面世的J2EE 1.4提供用Java开发Web应用程序的新的Servlet 2.4和JavaServer Pages (JSP) 2.0技术。本文展示了这两种技术的新特性,并在适当的地方提供每个特性的示例代码。本文假设读者认识以前的 Servlet 2.3和JSP 1.2版本。给出的例子已用Tomcat 5(包含在Java Web Services Developer Pack 1.2中)进行了测试。 Servlet和JSP毫无疑问是两种应用最广的J2EE技术。Servlet技术是用Java进行Web应用编程的基础,也是JSP的基础。但是,servlet编程可能会异常麻烦。特殊是当你不得不发送一个没多少代码的长HTML页面时更是如此。每个HTML 标记必须嵌入到字符串中,用PrintWriter对象的显示方法发送。是一种工作单调乏味而烦人的工作。使用servlet的另一个缺点是每一处改变都需要servlet程序员介入。 Sun公司了解到这一问题之后便开发了JSP作为解决方案。在JSP中,程序员和页面设计员的分工变得轻易多了,并且当JSP页面更改时会自动进行编译。不过请注重,JSP是servlet技术的一个扩展,而不是废弃servlet。在实际应用当中, servlet和JSP页面一起使用。 Servlet 2.4的新特性 Servlet 2.4提供了几个新类,且不支持javax.servlet.SingleThreadModel接口。这一版本只支持HTTP 1.1,所以Servlet 2.4应用程序不适用于HTTP 1.0客户程序。2.4版增加了哀求监听器和哀求属性监听器,并能在一个应用程序中将servlet用作欢迎页面。另外,Servlet 2.4还提供了更好的ServletRequest和RequestDispatcher对象,并更好地支持国际化。此外,现在是根据模式而不是文档类型定义(document-type definition,DTD)文件来验证部署描述符是否有效。这就意味着支持部署描述符的可扩展性。 下面详细说明Servlet 2.4的新特性。哀求监听器和哀求属性监听器。Servlet 2.3增加了servlet上下文相关监听器和会话相关监听器。Servlet 2.4增加了新的javax.servlet.ServletRequestListener和 javax.servlet.ServletRequestAttributeListener两种接口,它们会通知你与Request对象有关的事件。假如你对每个Request对象的初始化和撤消感爱好,你可以实施ServletRequestListener接口。这个接口有两个方式: requestInitialized()和requestDestroyed()。当需要一个Request对象时,servlet容器便调用 requestInitialized方式。当不再需要Request对象时,servlet容器便调用requestDestroyed方式。 这两个方式都从servlet容器接收一个javax.servlet.ServletRequestEvent对象。可以从ServletRequestEvent实例获得servlet上下文和servlet哀求。 第二个监听器接口ServletRequestAttributeListener处理Request对象属性的添加、更改和删除。该接口有以下方式: * attributeAdded。向Request对象添加新属性时由servlet容器调用。 * attributeRemoved。从Request对象中删除属性时由servlet容器调用。 * attributeReplaced。Request对象中现有属性值被替换时由servlet容器调用。 这三个方式从servlet容器获得 javax.servlet.ServletRequestAttributeEvent类的一个实例。 ServletRequestAttributeEvent类扩展了ServletRequestEvent类,并添加了两个新方式:getName和 getValue。getName方式返回触发事件的属性的名称,getValue返回属性的值。 代码清单1 给出这两个新的监听器的示例类。当servlet容器调用方式时二者都显示方式名。监听器经过编译后,它们的类文件必须被部署到WEB- INF/classes目录下。ServletRequest中的新方式。在Servlet 2.4中,javax.servlet.ServletRequest接口增加了4个新方式: * getRemotePort。返回发送哀求的客户机或最后一个代理服务器的Internet Protocol(IP)源端口。 * getLocalName。返回从中接收哀求的IP接口的主机名。 * getLocalAddr。返回从中接收哀求的接口的IP地址。 * getLocalPort。返回从中接收哀求的接口的IP端口号。 请注重,在Servlet 2.3中,getServerName和getServerPort方式返回的值就是现在getLocalName和getLocalPort返回的值。在2.4版中,getServerName和getServerPort已重新定义。欲了解更多的信息,请查看API文档。 将一个JSP页面中的代码示例如下-- out.println("<br>Remote Port : " + request.getRemotePort()); out.println("<br>Local Name : " + request.getLocalName()); out.println("<br>Local Addr : " + request.getLocalAddr()); out.println("<br>Local Port : " + request.getLocalPort()); --该代码生成这样的内容: Remote Port : 3303 Local Name : localhost Local Addr : 127.0.0.1 Local Port : 8080 哀求调度程序的新特性。使用哀求调度程序可将当前哀求传递给一个新的资源,或从当前页面引入另一个资源。Servlet 2.4增加了一些属性,它们将被添加到传递给另一个资源的一个Request对象上: javax.servlet.forward.request_uri javax.servlet.forward.context_path javax.servlet.forward.servlet_path javax.servlet.forward.path_info javax.servlet.forward.query_string 假如一个Request对象未被传递,则这些属性的值为null。另一方面,在所传递来对象的资源中这些属性将具有非null值。当某一个资源必须只能通过另一个资源调用而不能直接调用时,这些属性值很有用。 举个例子,在一个叫做myApp的Context(上下文)中有一个名为ModernServlet的servlet, ModernServlet被传递给TargetServlet。 在TargetServlet中,显示代码清单2中的代码。 myApp的部署描述符包含以下和元素: <servlet> <servlet-name>Modern</servlet-name> <servlet-class>ModernServlet </servlet-class> </servlet> <servlet-mapping> <servlet-name>Modern</servlet-name> <url-pattern>/Modern</url-pattern> </servlet-mapping> <servlet> <servlet-name>Target</servlet-name> <servlet-class>TargetServlet </servlet-class> </servlet> <servlet-mapping> <servlet-name>Target</servlet-name> <url-pattern>/Target</url-pattern> </servlet-mapping> 下面是调用ModernServlet时控制台显示的结果: javax.servlet.forward.request_uri : /myApp/Modern javax.servlet.forward.context_path : /myApp javax.servlet.forward.servlet_path : /Modern javax.servlet.forward.path_info : null javax.servlet.forward.query_string : null 将过滤器用于哀求调度程序。Servlet 2.4在部署描述符中添加了一个新的元素,以便servlet程序员决定是否将过滤器(filters)应用于哀求调度程序。元素的值可以是REQUEST(默认值)、FORWARD、INCLUDE和ERROR: * REQUEST。假如哀求直接来自客户机则使用过滤器。 * FORWARD。假如哀求正由哀求调度程序进行处理,表示与或相匹配的Web组件使用传递调用,则使用过滤器。 * INCLUDE。只有在哀求正由哀求调度程序进行处理,表示与或相匹配的Web组件使用包含(include)调用时,才使用过滤器。 * ERROR。只有在哀求正由错误页面机制处理为一个与元素相匹配的错误资源时才使用过滤器。 Servlet 2.4只支持HTTP 1.1客户机。Servlet 2.3既支持HTTP 1.0,又支持HTTP 1.1,而Servlet 2.4与Servlet 2.3不同,它只支持HTTP 1.1客户机。作为过渡,HTTP/1.0状态码302(暂时建议)仍旧存在,而且仍旧由 javax.servlet.http.HttpServletResponse接口中的SC_MOVED_TEMPORARILY表示。HTTP 1.1具有Found的状态码302,它由HttpServletResponse接口中的静态SC_FOUND表示。 Servlet用作欢迎页面。在Servlet 2.3中,你可以在部署描述符中使用元素列出欢迎文件--当收到一个不完整的URL时将显示的文件。但是,在Servlet 2.3中,在元素中只能使用HTML文件或JSP文件。在Servlet 2.4中,如今可以将一个servlet用作欢迎页面。下例为一个叫做Modern的servlet,它的类为ModernServlet.class,并已被映射到path /Modern。 <servlet> <servlet-name>Modern</servlet-name> <servlet-class>ModernServlet </servlet-class> </servlet> <servlet-mapping> <servlet-name>Modern</servlet-name> <url-pattern>/Modern</url-pattern> </servlet-mapping> <welcome-file-list> <welcome-file>Modern</welcome-file> </welcome-file-list> 此时,若用户键入诸如http://domain/context/(不带资源文件)的URL时,就会调用ModernServlet。 对国际化的新支持。在Servlet 2.3中,没有办法直接告诉客户浏览器应当使用什么字符编码。要实现这一目的,你必须把一个java.util.Locale对象传递给 javax.servlet.ServletResponse接口的setLocale方式,如下所示: response.setLocale(locale); 这意味着你必须首先创建一个Locale对象。 另外一种办法是,在Servlet 2.3中,你可以使用setContentType方式来传递内容类型和字符集,如: setContentType(\\\'text/html; charset=UTF-8\\\'); 在Servlet 2.4中,javax.servlet.ServletResponse接口中有两个支持国际化的新方式。第一个方式是setCharacterEncoding,它的用法如下: public void setCharacterEncoding(String charset) 使用setCharacterEncoding,你可以只将字符编码指定为一个字符串,而不必先创建Locale对象。不过,请注重,要让这种方式起作用,必须在调用getWriter方式之前以及响应提交之前调用它。 第二个新方式是getContextType,作为在ServletResponse对象中调用setContentType、setLocale或setCharacterEncoding方式的结果,它返回在ServletResponse对象中使用的内容类型。 除了javax.servlet.ServletResponse中的这两个方式之外,你还可以利用Servlet 2.4在部署描述符中定义一个新元素:它使servlet程序员不必在他/她的servlet中指定locale-to-charset映射。如何使用这一新元素的例子如下: <locale-encoding-mapping-list> <locale-encoding-mapping> <locale>ja</locale> <encoding>ISO-2022-JP</encoding> </locale-encoding-mapping> </locale-encoding-mapping-list> 部署描述符的可扩展性。在Servlet 2.3应用程序中,根据DTD文件对部署描述符进行验证。现在Servlet 2.4支持根据模式对部署描述符进行验证。使用模式比使用DTD有以下几点好处: * 通过模式可以继续另一个模式(可扩展的)的语法。 * 模式比DTD更精确。 * 通过模式可以指定每个元素的内容的实际数据类型。 * 模式可以用于多个名字空间。 * 通过模式可以指定一个元素出现的最多和最少次数。 但是,为了向后兼容,要求Servlet 2.4容器支持Servlet 2.3和Servlet 2.2 DTD。 不支持javax.servlet.SingleThreadModel接口。 SingleThreadModel接口没有方式,它用于向servlet容器指明,它必须保证不会有两个线程同时执行实施该接口的servlet的服务方式。从servlet技术开始出现到现在,人们普遍误解了这个接口。现在大家都反对用它,因为它会造成混乱,并且在考虑线程安全时在安全性方面给 servlet程序员一个错觉。在任何新的开发工作中决不应再使用这个接口。 JSP 2.0中的新特性 JSP 2.0(最初称为JSP 1.3)比JSP 1.2有了重要改进。当然,增加的最重要内容是JSP 2.0容器中加入了对表达式语言(EL)的支持。 EL最初是由JSP标准标记库(JSTL)1.0规范定义的,它可协助从JSP页面中删除Java 代码。javax.servlet.jsp.el包中所描述的API揭示EL的语义。EL表达式的语义与Java表达式的语义类似;表达式的值计算出来后被插入到当前的输出中。EL可用于标准的或定制的操作的属性值以及模板文本中。下面是EL表达式的结构(其中expr为表达式): ${expr} 对于包含字符序列"${"的文字值,JSP 2.0提供了一种方式,通过使用序列"${\\\'${\\\'"进行换码。例如,下面的字符序列被转变为文字值${expr}: ${\\\'${\\\'}expr} 此外,由于JSP 2.0以前的版本不支持EL,所以JSP应用程序将忽略任何Web应用程序中的EL,这些应用程序的web.xml根据Servlet 2.2或Servlet 2.3 DTD进行验证。为了测试此处讲到的JSP页面中的表达式,你只需从应用程序中删除web.xml文件。 实际上,EL是一种简朴的语言,它帮助页面创作者访问JSP隐含对象,进行反复操作以及不包含Java代码的条件操作--这些在JSP 1.2中是无法实现的。 为了访问隐含对象,JSP容器支持下面的名称-对象映射: * pageContext。PageContext对象 * pageScope。将页面范围的属性名映射到它们的值 * requestScope。将哀求范围的属性名映射到它们的值 * sessionScope。将会话范围的属性名映射到它们的值 * applicationScope。将应用程序范围的属性名映射到它们的值 * param。将参数名映射到一个单一串参数值 * paramValues。将参数名映射到该参数所有值的一个字符串数组 * header。将标头名映射到一个单一串标头值 * headerValues。将标头名映射到该标头所有值的一个字符串数组 * cookie。将cookie名映射到一个单一cookie对象 * initParam。将上下文初始化参数名映射到其字符串参数值 例如,下面的表达式表示参数userName的值: ${param.userName} 下面的表达式返回Session对象的productId属性的值: ${sessionScope.productId} 更简朴的SimpleTag接口操作过程。JSP 2.0提供了一个新的接口javax.servlet.jsp.tagext.SimpleTag,它是编写标记处理器(tag handler)的一种更简朴的方式。在JSP 1.2中,标记处理器必须直接或间接地实施avax.servlet.jsp.tagext包中的下列接口之一:Tag、IterationTag或 BodyTag。对于实施Tag接口的标记处理器来说,最基本的情况是,JSP容器每次碰到JSP页面中的一个标记时就调用doStartTag和 doEndTag两个方式。利用JSP 2.0,JSP程序员可以通过实施新的SimpleTag接口来选择实施过程更简朴的标记处理器。JSP容器并不调用实施Tag接口的标记处理器的两个方式,而只需要调用SimpleTag接口中的一个方式:doTag。所有标记逻辑、反复操作和主体评估等都用这一个方式来执行。所以,SimpleTag 与javax.servlet.jsp.tagext.BodyTag功能相同强盛,但操作过程更简朴。 为了支持需要实施SimpleTag接口的标记处理器的编写,javax.servlet.jsp.tagext包提供了一个名为SimpleTagSupport的支持类。假如你要扩展这个类,则你只需提供一个执行方式:doTag。 代码清单3给出了一个扩展SimpleTagSupport的标记处理器的例子。 使用标记文件更轻松地开发标记库。众所周知,JSP 1.2中的自定义标记库需要花很多时间来开发。开发工作涉及标记处理器和标记库描述符(TLD)文件的开发,以及标记库在web.xml文件中的注册。 JSP 2.0通过提供一种新的编写自定义标记库的方式解决了这个问题。使用标记文件,标记扩展可类似于JSP文件。无需编译,无需编辑web.xml文件,而且不再需要TLD。要做的是你必须把标记文件复制到WEB-INF/ tags目录中,而这一点很轻易做到。剩下的事都交给JSP容器去做,它会把WEB-INF/tags目录中找到的每个标记文件转变为标记处理器。程序员完全挣脱了构建标记处理器的复杂工作。 下面举个例子。这是标记库最简朴的形式,其中标记文件只是简朴地把一个字符串写到隐含对象中。 <%― example1.tag file, must reside in WEB-INF/tags ―%> <% out.println("Hello from tag file."); %> 使用JSP页面中的标记库再简朴不过了。和寻常相同,你只需taglib指令,通过前缀属性在整个页面中识别标记库。现在你有一个tagdir属性,而不是uri属性。tagdir属性引用WEB-INF/tags目录或WEB-INF/tags下的任何子目录。 下面是一个使用example1.tag文件的JSP页面的例子。 <%@ taglib prefix="easyTag" tagdir="/WEB-INF/tags" %> <easyTag:example1> </easyTag:example1> 调用该JSP页面浏览器上就会显示下面的字符串: Hello from tag file. 结合上面讲到的表达式语言,你就可以真正迅速构建无脚本的JSP页面。再举一个例子,下面的标记文件(叫做example2.tag)通过调用JSP页面接收一个属性,并将它转变为大写字母。 <%― example2.tag file, must reside in WEB-INF/tags ―%> <%@ attribute name="x" %> <% x = x.toUpperCase(); out.println(x); %> 下面是使用该标记文件的JSP页面: <%@ taglib prefix="easyTag" tagdir="/WEB-INF/tags" %> <easyTag:example2 x="hello"> </easyTag:example2> 下面是另一个例子,其中没有Java代码: <%― example3.tag file, must reside in WEB-INF/tags ―%> <%@ variable name-given="x" scope="AT_BEGIN" %> <%@ taglib prefix="c" uri="http://java.sun.com /jsp/jstl/core" %> <c:set var="x" value="3"/> After: ${x} <jsp:doBody/> 该标记文件用于下面的JSP页面: <%@ taglib prefix="c" uri="http://java.sun.com /jsp/jstl/core" %> <%@ taglib prefix="easyTag" tagdir="/WEB-INF/tags" %> <c:set var="x" value="1"/> Before: ${x}<br> <easyTag:example3/> 请注重,要运行本示例,在WEB-INF/lib目录下要有JSTL库。 最后一个标记文件示例还表明,不认识Java编程语言的页面创作者仍能利用标记扩展的强盛功能。即便是Java程序员,使用标记文件也比编写实施javax.servlet.jsp.tagext包中的某个接口的Java类要方便。 结论 本文简要阐述了Servlet 2.4和JSP 2.0规范中的新特性,它们将包含在即将面世的J2EE 1.4中。Servlet 2.4和JSP 2.0无疑将会加快Web应用程序的开发。 返回类别: 教程 上一教程: JSP简明教程:令人高兴的脚本编程 下一教程: 利用WEBLOGIC的POOL(连接池)连接数据库 您可以阅读与"SERVLET和JSP迈上新台阶"相关的教程: · SERVLET和JSP的通信的一种方式 · 入门必读:SERVLET/JSP配置超详解 · 选择jsp而不是servlet作为BS前台主流方案是JAVA的战略性方向错误 · APACHE+SERVLET+JSP环境设置(中) · Apache+Servlet+Jsp环境设置(下) |
![]() ![]() |
快精灵印艺坊 版权所有 |
首页![]() ![]() ![]() ![]() ![]() ![]() ![]() |