本文是三篇学习struts框架系列文章的最后一篇(原文请见http://www.onjava.com/pub/a/onjava/2001/11/14/jsp_servlets.html)。
在第一篇文章《jakarta struts简介》中,我大致分析了struts框架,讨论了它所能完成的功能,还浏览了组成struts的各个组成部分。在第二篇文章《学习jakarta struts》中,我开始详细描述如何利用struts来构建一个简单应用的过程步骤。而本篇文章将会向大家演示如何将applicationresource文件中的文本信息,通过struts标签在jsp页面中显示出来。
action类是连接struts架构和应用中业务逻辑代码的桥梁。所以你应该尽可能让action类小巧简单,因为真实应用中的逻辑处理应该是由单独分离出来的逻辑层来完成的。如果你正在从事n层应用的开发,你当然希望层与层之间的接口越简单越好。而事实上,action类中的主要方法"perform()"(1.1中为execute())却有点暗示应该在本方法中做点什么的意思。我们知道,每个action类都需要从 org.apache.struts.action.action 继承而来。在小型应用中,我们的action类很可能就只要继承org.apache.struts.action.action就足够了;而在某些特定的复杂应用中,我就从我们所实现的action类中总结出来了一些通用特性。因此,在我看来,构造一个基类将这些通用特性的代码实现出来,让应用中所用到的所有action类不直接继承org.apache.struts.action.action,而继承这个完成了一些通用特性的基类以实现代码重用,是一个相当不错的设计。我在strutssample中就应用了这种方法,构造了这样的一个基类,该基类的方法在完成复杂逻辑的和简单转发请求的action类中都可以使用。
package com.oreilly.actions;
import java.io.ioexception;
import java.util.properties;
import java.util.resourcebundle;
import java.util.missingresourceexception;
import java.util.enumeration;
import java.util.properties;
import java.rmi.remoteexception;
import javax.ejb.ejbhome;
import javax.ejb.createexception;
import javax.naming.context;
import javax.naming.initialcontext;
import javax.naming.namingexception;
import javax.servlet.servletexception;
import javax.servlet.http.httpservletrequest;
import javax.servlet.http.httpservletresponse;
import org.apache.struts.action.action;
import org.apache.struts.action.actionservlet;
import org.apache.struts.action.actionform;
import org.apache.struts.action.actionforward;
import org.apache.struts.action.actionmapping;
这个类就是使用struts开发时,所有action类都要继承的基类。它把一些通常在实际应用中最有可能被用到的东西都考虑进来了。就这篇文章而言, 类中一些与struts并不是太紧密相关的方法将只做注释而不会完整的实现,而从事开发工作的你,有兴趣的话,请完成这些方法并应用这个类,将为你在实际项目中的开发快马加鞭。注意,因为所有的action类都要从org.apache.struts.action.action 继承而来,所以我们的这个类同样。
public abstract class abststrutsactionbase extends action {
/ * 定义一些在struts-config.xml中记录在案的
* 全局应用中皆可可通用的forward标识*/
protected static final string success = "success";
protected static final string failure = "failure";
protected static final string error = "error";
protected static final string login = "login";
protected static final string confirm = "confirm";
protected context jndicontext = null;
/**
* 默认构造方法
*/
public abststrutsactionbase() {
}
/**
下面这个查找ejb实例的方法将不会完整实现。
一般来说,action类应该调用实现了应用的商务逻辑的ejb会话bean(或仅仅普通javabean)。在大型项目中,开发人员必须划清层与层之间的界限。在action类中,我们应该拿到获取含有jndi信息的环境的实例,然后通过ejb的jndi名字去查询获取它的home接口。过程并不简单,所以下面这个代码片断只是个给出了必要实现的小例子。
 参数类型string,传入的要查询jndi的名字
 返回类型object,即查找到的home接口
 如果查找失败,抛出namingexception异常
 如果获取资源信息失败,抛出missingresourceexception异常
*/
public object lookup(string jndiname)
throws namingexception, missingresourceexception {
// 为调用ejb对象,通过构建记录jndi信息的properties对象
// 来获得初始环境信息
if (jndicontext == null) {
resourcebundle resource =
resourcebundle.getbundle("strutssample.properties");
properties properties = new properties();
properties.setproperty(
context.initial_context_factory,
resource.getstring(context.initial_context_factory));
properties.setproperty(
context.provider_url,
resource.getstring(context.provider_url));
properties.setproperty(
context.security_principal,
resource.getstring(context.security_principal));
properties.setproperty(
context.security_credentials,
resource.getstring(context.security_credentials));
jndicontext = new initialcontext(properties);
}
注意:在真正的产品中,我们应该在此处考虑代码的健壮性,将代码加入到try/catch块内,并记录所有错误或重要信息到系统log中。而本例中,我们仅仅把异常往外抛,并假定一定会找到ejb对象的home接口并返回。
return (jndicontext.lookup(jndiname));
}
由于action类将是由struts来调用的。所以它的主要方法应该是一个抽象方法,而由每个继承的子类来具体实现,或者在其中做一些所有action都会做的通用机制,例如记录log信息。在本例中,我们一切从简,将其抽象之。
 参数mapping:其类型为actionmapping,将在本action做跳转选择用
 参数actionform:由struts根据本次http请求数据填充完成的actionform对象(可选,如果存在请求数据的话)
 参数request:此action所有处理的本次http请求(对象)
 参数response:此action输出数据所要用到的http响应(对象)
 如果有i/o错误出现,则本方法抛出ioexception异常
 如果处理时发生servlet异常,则本方法抛出servletexception异常
 本方法处理完请求后按照处理逻辑返回相应的页面导向(对象)
public abstract actionforward perform(
actionmapping mapping,
actionform form,
httpservletrequest request,
httpservletresponse response)
throws ioexception, servletexception;
}
或者让这个抽象方法更有用一点,那就在里面干点什么吧,比如像下面这样在其中记录log。
{
actionforward forward = null;
// 只是简单的记录一些提示信息到servlet log
getservlet().log(
"abststrutsactionbase.perform() [action class: "
+ this.getclass().getname()
+ " ]");
getservlet().log(
"abststrutsactionbase.perform() [form class : "
+ (form == null ? "null" : form.getclass().getname())
+ " ]");
}
然后,我们再编写的每个action类都应该从abststrutsactionbase继承,并依照处理逻辑编写各自的perform方法。让我们用loginaction为例,看看具体应该怎么应用吧。
package com.oreilly.actions;
import java.io.ioexception;
import java.rmi.remoteexception;
import javax.ejb.createexception;
import javax.servlet.servletexception;
import javax.servlet.http.httpservletrequest;
import javax.servlet.http.httpservletresponse;
import org.apache.struts.action.actionerror;
import org.apache.struts.action.actionerrors;
import org.apache.struts.action.actionform;
import org.apache.struts.action.actionmapping;
import org.apache.struts.action.actionforward;
import com.oreilly.forms.loginform;
/*
loginaction 将演示一个action将如何被struts架构所调用
在这个例子中,我们只是简单的演示perform方法是如何调用、执行并返回的
*/
public class loginaction extends abststrutsactionbase {
接下来这个是验证用户的方法,本例中没有具体实现。但一个典型的应用方案是调用javabean或者ejb来完成。用来查找ejb的lookup方法(在基类中完成的)应该在本方法中被调用,其返回一个依据后台数据库验证用户的接口。
 参数类型string,要验证的用户名
 参数类型string,密码
 返回类型boolean,如果验证通过为true,否则为false
public boolean authenticate(string username, string password) {
/* 本方法将先做一个查找动作,获得验证用户的ejb对象的接口并调用
* 由于本例只演示action与商务逻辑层是如何交互的
* 所以具体实现代码本例中就不提供了:)
*/
return (true);
}
接下来我们在loginaction中重载基类的perform方法。
 参数mapping:其类型为actionmapping,将在本action做跳转选择用
 参数actionform:由struts根据本次http请求数据填充完成的actionform对象(可选,如果存在请求数据的话)
 参数request:此action所有处理的本次http请求(对象)
 参数response:此action输出数据所要用到的http响应(对象)
 如果有i/o错误出现,则本方法抛出ioexception异常
 如果处理时发生servlet异常,则本方法抛出servletexception异常
 本方法处理完请求后按照处理逻辑返回相应的页面导向(对象)
public actionforward perform(
actionmapping mapping,
actionform form,
httpservletrequest request,
httpservletresponse response)
throws ioexception, servletexception {
// 先假定验证失败,那么要导向的forward当然是login了(见基类定义的全局变量)
boolean validlogin = false;
actionforward actionforward = mapping.findforward(login);
// 构造出承载actionerror对象的容器——errors,以备错误出现时可用
actionerrors errors = new actionerrors();
// 从由本次请求构造的actionform中提取出所需要的数据
loginform loginform = (loginform)form;
string username = null;
string password = null;
if (loginform != null) {
username = loginform.getusername();
password = loginform.getpassword();
validlogin = authenticate(username, password);
}
if (validlogin) {
// 验证成功了,导向到struts-config.xml中定义的success
actionforward = mapping.findforward(success);
// 存点必要的东东到session,以备后用
request.getsession(true).setattribute("username", username);
} else {
errors.add("login", new actionerror("error.login.authenticate"));
}
// 系统如果用户界面友好一点,我们就应该将错误信息存入request对象中
// 然后到页面,通过在struts的标签显示出来
if (!errors.empty()) {
saveerrors(request, errors);
}
// 本action处理完成,导向到合适的forward
return (actionforward);
}
}
注意,这个loginaction类就是在struts-config.xml中定义的用来处理登录事务的一个具体实现。当这个类被载入并有一个对象实例化后,struts架构就会调用它的perform方法。这个方法是这样声明的:
public actionforward perform(
actionmapping mapping,
actionform form,
httpservletrequest request,
httpservletresponse response)
throws ioexception, servletexception
其中,mapping是一个记录与此action对应的forward导向的对象,form对象封装由客户端提交的此action要处理的数据,还有标准的httpservletrequest对象和httpservletresponse对象。
有了这些对象的辅助,此action就可以拿到需要的东东顺利开工了。我们的例子中,要处理的数据主要是用户名和密码,这些都由form对象提供。实现验证功能是本应用的主要业务逻辑,在方法中的具体实现应该是去取ejb的相应接口来操作或者直接去拿数据库数据来验证。前面那个abststrutsactionbase类中已经实现了一个简单的ejb接口查找动作,所以如果我们是在开发一个基于ejb实现的系统,它的可重用性就非常强了。
由验证方法(authenticate())的返回值,action要接着做出合理的动作。如果验证通过,就要让用户进入正确的页面,那么我们就将一些后面可能会用到的信息存入request对象(译注:准确的讲,代码中是存到了session对象里,当然session对象是和当前request相关的),并向struts返回success这个forward。这个forward是在struts-config.xml中定义的,然后由actionmapping封装起来,在action处理中可以从中拿出合适的forward做为返回值。如果回头去看看struts-config.xml中的定义,就会知道success这个forward会将用户导向至welcome.jsp这个页面的。如果验证失败,则将一个错误信息存起来,然后导向到一个错误提示页面显示出来。
开发应用的业务逻辑
在一个真实的应用系统中,我们应该将业务逻辑层整合进来了。在我们这个例子里,我们就应该去开发loginaction中的authenticae方法所调用到的ejb了。但是正如你所见的,我们完全可以把这一层暂时屏蔽掉,而利用struts把前端部分构建并能够让它跑起来的。我其实相当推崇的是方法是先将应用框架搭建并运行起来,然后在开发后台实际的业务逻辑层。在应用框架完全恰当的构建起来的时候,后台的开发工作所有做的debug工作也少的多了。而且,业务逻辑的开发也不是本文所要函概的范围,所以此处我们略过,不过我相信你现在一定对应用的全局有了总体的把握了吧!
开发由actionmapping定义的系统工作流程,完成对应的jsp页面
终于可以将所有这些东东整合在一起了。在struts-config.xml配置文件中定义的那些actionmapping,我们要完成这些actionmapping定义用到的jsp页面。本例中,包括login.jsp、welcome.jsp和errorpage.jsp。还有,尽管我们在本例中都是将action处理完成forward到jsp页面,这在这个简单的例子中是再恰当不过的逻辑流程了,而在实际利用struts开发应用中呢,当然可以从action forward到其他的action。我们这个简单的login.jsp页面内容是这样的:
<%@ page language="java" %>
<%@ taglib uri="/web-inf/struts-html.tld" prefix="html" %>
<%@ taglib uri="/web-inf/struts-form.tld" prefix="form" %>
<%@ taglib uri="/web-inf/struts-bean.tld" prefix="bean" %>
<html>
<head>
<title><bean:message key="login.title"/></title>
</head>
<body>
<html:errors/>
<h3>enter your username and password to login:</h3>
<html:form action="login.action" focus="username" >
<html:text property="username" size="30" maxlength="30"/>
<html:password property="password" size="16" maxlength="16" redisplay="false"/>
<html:submit property="submit" value="submit"/>
<html:reset/>
</html:form>
</body>
</html>
<html>
<title>welcome to struts</title>
<body>
<p>welcome <%= (string)request.getsession().getattribute("username") %></p>
</p>you have logged in successfully!</p>
</body>
</html>
<?xml version="1.0" encoding="iso-8859-1"?>
<!--
this is the web-app configuration that allow the strutssample to work under
apache tomcat.
-->
<!doctype web-app
public "-//sun microsystems, inc.//dtd web application 2.2//en"
"http://java.sun.com/j2ee/dtds/web-app_2_2.dtd";>
<web-app>
<servlet>
<servlet-name>oreilly</servlet-name>
<servlet-class>org.apache.struts.action.actionservlet</servlet-class>
<init-param>
<param-name>application</param-name>
<param-value>com.oreilly.applicationresources</param-value>
</init-param>
<init-param>
<param-name>config</param-name>
<param-value>/web-inf/struts-config.xml</param-value>
</init-param>
<init-param>
<param-name>debug</param-name>
<param-value>2</param-value>
</init-param>
<init-param>
<param-name>detail</param-name>
<param-value>2</param-value>
</init-param>
<init-param>
<param-name>validate</param-name>
<param-value>true</param-value>
</init-param>
<load-on-startup>2</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>oreilly</servlet-name>
<url-pattern>*.action</url-pattern>
</servlet-mapping>
<welcome-file-list>
<welcome-file>login.jsp</welcome-file>
</welcome-file-list>
<!-- struts tag library descriptors -->
<taglib>
<taglib-uri>/web-inf/struts.tld</taglib-uri>
<taglib-location>/web-inf/struts.tld</taglib-location>
</taglib>
<taglib>
<taglib-uri>/web-inf/struts-bean.tld</taglib-uri>
<taglib-location>/web-inf/struts-bean.tld</taglib-location>
</taglib>
<taglib>
<taglib-uri>/web-inf/struts-html.tld</taglib-uri>
<taglib-location>/web-inf/struts-html.tld</taglib-location>
</taglib>
<taglib>
<taglib-uri>/web-inf/struts-logic.tld</taglib-uri>
<taglib-location>/web-inf/struts-logic.tld</taglib-location>
</taglib>
<taglib>
<taglib-uri>/web-inf/struts-form.tld</taglib-uri>
<taglib-location>/web-inf/struts-form.tld</taglib-location>
</taglib>
</web-app>
Java Asp PHP .Net XML C/C++ CGI VB Jsp J2ee J2se J2me EJB Servlet Tomcat Resin Struts Weblogic Eclipse ANT GUI JMS Web servise IDEA Webphere Hibernate Spring Jboss Applet Swing Socket Javamail Perl Ajax P2P 安全 模式 框架 测试 开源 游戏
Windows XP Windows 2000 Windows 2003 Windows Me Windows 9.x Linux UNIX 注册表 操作系统 服务器 应用服务器