| |
javaserver pages(jsf) 在 java 的 web 编程中已经被认为是下一个重大的事件。通过 jsf ,你可以在网页上使用 web 组件,来捕获用户行为产生的事件。不远的将来,java 工具将支持这个技术,开发 web 应用程序将与我们现在开发 swing 程序类似:拖放控件、写事件侦听器。本文是一个 jsf 的简要介绍,并且提供一个 jsf 的例子,用来展示 jsf 的事件驱动特性。要理解本文,您需要对 servlets, jsp, javabeans, 与标签库有一定的理解。 首先,一个 jsf 应用就是一个 servlet/jsp 应用。它有一个配置描述符,有 jsp 页面、客户定制标签、静态资源等等。不同的是,jsf 应用是事件驱动的。你通过写一个事件侦听类来决定应用程序的行为。以下建立一个 jsf 应用所需要的几个步骤: 1、建立 jsp 页面,用 jsf 组件包装 html 元素。 2、写一个 javabean 用来保持用户输入与组件数据的状态。 3、写一个事件侦听器来决定当某事件发生时应该有什么反映,比如用户点击了一个按钮或者提交了表单。jsf 支持两个事件:actionevent 与 valuechangeevent 。actionevent 是针对用户提交表单与点击按钮的,而 valuechangeevent 是当一个 jsf 组件改变了时触发。 现在,让我们来看一下 jsf 动作的细节。 jsf 怎样工作 jsp 页面是 jsf 应用的用户接口。每个页面包括一些 jsf 组件用来描述 web 控件,如表单、输入框、按钮等等。组件可以嵌入另一个组件中,正如输入框可以在表单中。每个 jsp 页面就这样表示为组件树。javebeans 从用户的请求中获取数据并存储。 这是有意思的部分:每当用户做任何事情,如点击按钮或者提交表单,都有事件产生。然后事件消息通过 http 传到服务器。在服务器端,是一个配置了叫做 faces servlet 的特殊 servlet 的 web 容器。faces servlet(javax.faces.webapp.facesservlet)是所有 jsf 应用的引擎。每个 jsf 应用在 web 容器中都有独立的 faces servlet 。另一个重要的对象是 javax.faces.context.facescontext , 它包括了所有关于当前用户请求的必要信息。 faces servlet 的后台处理是相当复杂的。然而你没有必要了解这些细节,只需要记住:faces servlet 为 jsp 页面创建了组件树,对组件树的控制又对应着事件。faces servlet 知道怎么去创建组件树,因为它已经访问了当前应用中所有的 jsp 页面。faces servlet 还会创建一个 event 对象,并把它传递给所有注册过的侦听器。你可以通过与当前请求相对应的 facescontext 得到这个页面的组件树。 客户端浏览器上 web 控件产生的事件,被包含在一个 http 请求中,放在一起还有如浏览器类型、请求地址等其它信息。因此,所有需要 faces servlet 处理的请求必须指向这个 servlet 。那你怎样通过调用 faces servelt 来处理每个 http 请求呢?很容易,只需要在配置描述符里用一个 servlet-mapping 元素把一个特殊的 url 式样映射到 faces servlet。通常,你会用到 /faces/* 样式,如下所示: <!-- faces servlet --> <servlet> <servlet-name>faces servlet</servlet-name> <servlet-class>javax.faces.webapp.facesservlet</servlet-class> <load-on-startup>1</load-on-startup> </servlet> <!-- faces servlet mapping --> <servlet-mapping> <servlet-name>faces servlet</servlet-name> <url-pattern>/faces/*</url-pattern> </servlet-mapping> 请求地址必须包含有在 <url-pattern> 元素中描述的样式。这个要求不容易达到。另外也需要注意的是 <servlet> 元素,它包含 faces servlet ,有一个 <load-on-startup> 元素,用来确是否应用程序第一次启动时 servlet 是否加载。 为了捕获组件产生的事件,你需要为这个组件写一个侦听器,并把它注册给这个组件。通过在表示组件的客户端标签中嵌入 <action_listener> 元素能做到这一点。例如,为了让一个名叫 jsfapp.myactionlistener 的事件侦听器,来捕获一个名叫 submitbutton 的命令按钮产生的事件,在你的 jsp 页面中写如下的代码即可: <h:command_button id="submitbutton" label="add" commandname="submit" > <f:action_listener type="jsfapp.myactionlistener" /> </h:command_button> 一个 action listener 必须实现 javax.faces.event.actionlistener 接口,而一个 value-changed listener 必须实现 java.faces.event.valuechangedlister 接口。下面让我们来创建一个简单的 jsf 应用,以展现 jsf 是怎么样事件驱动的。 一个简单的 jsf 应用 我们将创建一个简单的应用,它可以实现对二个数字相加。为了运行这个应用,你需要准备 tomcat5 与 jsf v1.0 ea4(包含在 java web services developer pack (jwsdp) 1.2中)。这个应用程序包括: adder.jsp jsp 页面。 numberbean 存放用户数据的 javabean myactionlistener 事件侦听器 web.xml 配置描述文件 为了使这个应用能正常工作,还需要几个 jar 文件,包括 jsf 标准实现与其它类库。如果你安装了 jwsdp 1.2,你就可以在 jsflib 目录下找到所需要的这些文件。把这些 .jar 文件拷贝到 web-inf/lib 目录下。下面是整个的 .jar 与 .tld 文件列表: jsf-api.jar 包含有 faces servlet 与其它相关 javax.faces 包下面的类 jfs-ri.jar 是 jsf 的参考实现 jstl_el.jar standard.jar 此外,一个 jsf 的应用还需要如下的类库,它们是 apache jakarta 项目的一部分: commons-beanutils.jar commons-digester.jar commons-logging.jar is 以下的几小段讨论这个 jsf 示例的每个部分。最后的一小段,“编译与运行”,解释 jsf 应用怎么样运行。 创建目录结构 首先为你的 jsf 应用创建一个目录结构。在 tomcat 中,它在 webapps 目录下。“图1”描述了叫做 myjsfapp 的应用程序的目录结构。 写配置描述符 与其它的 servlet/jsp 应用一样,这个应用程序也需要一个配置描述文件。如“清单1”表示。 listing 1. the deployment descriptor (the web.xml file) <?xml version="1.0"?> <!doctype web-app public "-//sun microsystems, inc.//dtd web application 2.3//en" "http://java.sun.com/dtd/web-app_2_3.dtd";;> <web-app> <!-- faces servlet --> <servlet> <servlet-name>faces servlet</servlet-name> <servlet-class>javax.faces.webapp.facesservlet</servlet-class> <load-on-startup> 1 </load-on-startup> </servlet> <!-- faces servlet mapping --> <servlet-mapping> <servlet-name>faces servlet</servlet-name> <url-pattern>/faces/*</url-pattern> </servlet-mapping> </web-app> 在这个配置描述文件中有二个部分。 <servlet> 元素注册 faces servlet , <servlet-mapping> 元素声明任何包含有 /faces/ 式样的请求地址,必须传递给 faces servlet 。 创建 jsp 页面 一个叫做 adder.jsp 的 jsp 页面提供用户接口,如“清单2”所示: listing 2. the adder.jsp page <%@ taglib uri="http://java.sun.com/jsf/html";; prefix="h" %> <%@ taglib uri="http://java.sun.com/jsf/core";; prefix="f" %> <html> <head> <title>add 2 numbers</title> </head> <body> <jsp:usebean id="numberbean" class="jsfapp.numberbean" scope="session" /> <f:use_faces><br /> <h:form id="addform" formname="addform" ><br /> first number:<br /> <h:input_number id="firstnumber" valueref="numberbean.firstnumber" /><br /> second number: <h:input_number id="secondnumber" valueref="numberbean.secondnumber" /><br /> result: <h:output_number id="output" valueref="numberbean.result"/><br> <h:command_button id="submitbutton" label="add" commandname="submit" > <f:action_listener type="jsfapp.myactionlistener" /> </h:command_button> </h:form> </f:use_faces> </body> </html> 我们首先定义了俩个标签,它用到 jsf 的两个标签库:html 与 core 。这俩个标签库的定义可以在 jsf-ri.jar 文件中找到,所以你不用为它担心。它们的前缀分别是 h / f 。 <%@ taglib uri="http://java.sun.com/jsf/html";; prefix="h" %> <%@ taglib uri="http://java.sun.com/jsf/core";; prefix="f" %> <jsp:usebean> 这个动作元素定义 numberbean javabean 为 session scope 。 <jsp:usebean id="numberbean" class="jsfapp
|
|