有struts 1.x经验的朋友都知道action是struts的核心内容,当然struts 2.0也不例外。不过,struts 1.x与struts 2.0的action模型很大的区别。
|
struts 1.x |
stuts 2.0 |
接口 |
必须继承org.apache.struts.action.action或者其子类 |
无须继承任何类型或实现任何接口 |
表单数据 |
表单数据封装在formbean中 |
表单数据包含在action中,通过getter和setter获取 |
虽然,理论上struts 2.0的action无须实现任何接口或继承任何类型,但是,我们为了方便实现action,大多数情况下都会继承com.opensymphony.xwork2.actionsupport类,并重载(override)此类里的string execute()方法。具体的实现,如例1所示:
<%
@ page contenttype
=
"
text/html; charset=utf-8
"
%>
<%
@ taglib prefix
=
"
s
"
uri
=
"
/struts-tags
"
%>
<
html
>
<
head
>
<
title
>
hello world!
title
>
head
>
<
body
>
<
h2
><
s:property
value
="message"
/>
h2
>
body
>
html
>
例1 helloworld.jsp
package
tutorial;
import
java.text.dateformat;
import
java.util.date;
import
com.opensymphony.xwork2.actionsupport;
public
class
helloworld
extends
actionsupport
{
private
string message;
public
string getmessage()
{
return
message;
}
@override
public
string execute()
{
message
=
"
hello world, now is
"
dateformat.getinstance().format(
new
date());
return
success;
}
}
例1 classes/tutorial/helloworld.java
<
package
name
="actiondemo"
extends
="struts-default"
>
<
action
name
="helloworld"
class
="tutorial.helloworld"
>
<
result
>
/helloworld.jsp
result
>
action
>
package
>
例1 classes/struts.xml中helloworld action的配置
在浏览器地址栏中键入,可以看到如图1所示页面。
图1 helloworld输出页面
参考javadoc,可知类实现了接口:
- com.uwyn.rife.continuations.continuableobject
默认情况下,当请求helloworld.action发生时,struts运行时(runtime)根据struts.xml里的action映射集(mapping),实例化tutoiral.helloworld类,并调用其execute方法。当然,我们可以通过以下两种方法改变这种默认调用。这个功能(feature)有点类似struts 1.x中的lookupdispathaction。
- 在classes/sturts.xml中新建action,并指明其调用的方法;
- 访问action时,在action名后加上“!xxx”(xxx为方法名)。
实现方法请参考例2:
在classes/tutorial/helloworld.java中加入以下方法:
public string aliasaction() {
message ="自定义action调用方法";
return success;
}
例2 classes/tutorial/helloworld.java代码片段实现方法一,在classes/sturts.xml中加入下面代码:
<action name="aliashelloworld" class="tutorial.helloworld" method="aliasaction">
<result>/helloworld.jspresult>
action>
例2 classes/struts.xml中alaishelloworld action的配置实现方法二,使用地址来访问helloworld action。
在浏览器地址栏中键入或 ,可以看到如图2所示页面。
图2 自定义action调用方法页面通过上面的两个例子,细心的朋友应该可能会发现classes/tutorial/helloworld.java中action方法(execute和aliasaction)返回都是success。这个属性变量我并没有定义,所以大家应该会猜到它在actionsupport或其父类中定义。没错,success在接口com.opensymphony.xwork2.action中定义,另外同时定义的还有, , , 。
此外,我在配置action时都没有为result定义名字(name),所以它们默认都为success。值得一提的是struts 2.0中的result不仅仅是struts 1.x中forward的别名,它可以实现除forward外的很激动人心的功能,如将action输出到freemaker模板、velocity模板、jasperreports和使用xsl转换等。这些都过result里的type(类型)属性(attribute)定义的。另外,您还可以自定义result类型。
下面让我们来做一个velocity模板输出的例子,首先在classes/struts.xml中新建一个action映射(mapping),将其result类型设为velocity,如以下代码所示:
<action name="vmhelloworld" class="tutorial.helloworld">
<result type="velocity">/helloworld.vmresult>
action>
例3 classes/struts.xml中vmhelloworld action的配置新建helloworld.vm,内容如下所示:
<html>
<head>
<title>velocitytitle>
<meta http-equiv="content-type" content="text/html; charset=utf-8">
head>
<body>
<h2>message rendered in velocity: $messageh2>
body>
html>
例3 helloworld.vm在浏览器地址栏中键入,页面输出如下图3所示。
图3 helloworld.vm的输出页面 | 要运行例3需要在web-inf/lib中添加以下几个包:
|
前面,我花了不少的时间讨论action的输出。我老板有句名言——程序无非就是输入、操作和输出。因此,现在我们要讨论一下输入——表单输入。
使用struts 2.0,表单数据的输入将变得非常方便,和普通的pojo一样在action编写getter和setter,然后在jsp的ui标志的name与其对应,在提交表单到action时,我们就可以取得其值。
让我们看一个例子,新建login action,它通过login.jsp的表单获得用户名和密码,验查用户名是否为“max”,密码是否则为“secret”。如果,两者都符合,就在helloworld中显示“welcome, max”,否则显示“invalid user or password”。
package tutorial;
import com.opensymphony.xwork2.actionsupport;
publicclass login extends actionsupport {
private string name;
private string password;
private string message;
public string getname() {
return name;
}
publicvoid setname(string name) {
this.name = name;
}
public string getpassword() {
return password;
}
publicvoid setpassword(string password) {
this.password = password;
}
public string getmessage() {
return message;
}
@override
public string execute() {
if("max".equals(name) &&"secret".equals(password)) {
message ="welcome, " name;
}else{
message ="invalid user or password";
}
return success;
}
} 例4 classes/tutorial/login.java<%@ page contenttype="text/html; charset=utf-8" %>
<%@ taglib prefix="s" uri="/struts-tags"%>
<html>
<head>
<title>logintitle>
head>
<body>
<s:form action="login" method="post">
<s:textfield name="name" label="user name"/>
<s:password name="password" label="password"/>
<s:submit value="submit"/>
s:form>
body>
html>
例4 login.jsp<action name="login" class="tutorial.login">
<result>/helloworld.jspresult>
action>
例4 classes/struts.xml中login action的配置运行tomcat,在浏览器地址栏中键入,出现如图4所示页面。
图4 login.jsp输出页面分别在user name中输入“max”和“secret”,点击“submit”按钮,出现如图5所示页面。
图5 login成功页面在浏览器地址栏中键入,分别在user name中输入“scott”和“password”,点击“submit”按钮,出现如图6所示页面。
图6 login失败页面
struts 2.0更厉害的是支持更高级的pojo访问,如user.getpassword()。我们可以用另一写法实现例4。首先,将name和password从login类中分离出来,到新建类user中。这样对我们开发多层系统尤其有用。它可以使系统结构更清晰。
package tutorial;
import com.opensymphony.xwork2.actionsupport;
publicclass loginx extends actionsupport {
private user user;
private string message;
publicvoid setuser(user user) {
this.user = user;
}
public user getuser() {
return user;
}
public string getmessage() {
return message;
}
@override
public string execute() {
if("max".equals(user.getname()) &&"secret".equals(user.getpassword())) {
message ="welcome, " user.getname();
}else{
message ="invalid user or password";
}
return success;
}
} 例5 classes/tutorial/loginx.java<%@ page contenttype="text/html; charset=utf-8" %>
<%@ taglib prefix="s" uri="/struts-tags"%>
<html>
<head>
<title>logintitle>
head>
<body>
<s:form action="loginx" method="post">
<s:textfield name="user.name" label="user name"/>
<s:password name="user.password" label="password"/>
<s:submit value="submit"/>
s:form>
body>
html>
例5 loginx.jsp<action name="loginx" class="tutorial.loginx">
<result>/helloworld.jspresult>
action>
例5 classes/struts.xml中的loginx action配置
很多时候我的同事会问我:“如果我要取得servlet api中的一些对象,如request、response或session等,应该怎么做?这里的execute不像struts 1.x的那样在参数中引入。”开发web应用程序当然免不了跟这些对象打交道。在strutx 2.0你可以有两种方式获得这些对象:非ioc(控制反转inversion of control)方式和ioc方式。
非ioc方式
要获得上述对象,关键struts 2.0中类。我们可以通过它的静态方法getcontext()获取当前action的上下文对象。 另外,作为辅助类(helper class),可以帮助您快捷地获得这几个对象。
- httpservletrequest request = servletactioncontext.getrequest();
- httpservletresponse response = servletactioncontext.getresponse();
- httpsession session = request.getsession();
如果你只是想访问session的属性(attribute),你也可以通过actioncontext.getcontext().getsession()获取或添加session范围(scoped)的对象。
ioc方式
要使用ioc方式,我们首先要告诉ioc容器(container)想取得某个对象的意愿,通过实现相应的接口做到这点。具体实现,请参考例6 iocservlet.java。
package tutorial;
import javax.servlet.http.httpservletrequest;
import javax.servlet.http.httpservletresponse;
import javax.servlet.http.httpsession;
import org.apache.struts2.servletactioncontext;
import com.opensymphony.xwork2.actioncontext;
import com.opensymphony.xwork2.actionsupport;
publicclass noniocservlet extends actionsupport {
private string message;
public string getmessage() {
return message;
}
@override
public string execute() {
actioncontext.getcontext().getsession().put("msg", "hello world from session!");
httpservletrequest request = servletactioncontext.getrequest();
httpservletresponse response = servletactioncontext.getresponse();
httpsession session = request.getsession();
stringbuffer sb =new stringbuffer("message from request: ");
sb.append(request.getparameter("msg"));
sb.append("
response buffer size: ");
sb.append(response.getbuffersize());
sb.append("
session id: ");
sb.append(session.getid());
message = sb.tostring();
return success;
}
} 例6 classes/tutorial/noniocservlet.javapackage tutorial;
import java.util.map;
import javax.servlet.http.httpservletrequest;
import javax.servlet.http.httpservletresponse;
import javax.servlet.http.httpsession;
import org.apache.struts2.interceptor.servletrequestaware;
import org.apache.struts2.interceptor.servletresponseaware;
import org.apache.struts2.interceptor.sessionaware;
import com.opensymphony.xwork2.actioncontext;
import com.opensymphony.xwork2.actionsupport;
publicclass iocservlet extends actionsupport implements sessionaware, servletrequestaware, servletresponseaware {
private string message;
private map att;
private httpservletrequest request;
private httpservletresponse response;
public string getmessage() {
return message;
}
publicvoid setsession(map att) {
this.att = att;
}
publicvoid setservletrequest(httpservletrequest request) {
this.request = request;
}
publicvoid setservletresponse(httpservletresponse response) {
this.response = response;
}
@override
public string execute() {
att.put("msg", "hello world from session!");
httpsession session = request.getsession();
stringbuffer sb =new stringbuffer("message from request: ");
sb.append(request.getparameter("msg"));
sb.append("
response buffer size: ");
sb.append(response.getbuffersize());
sb.append("
session id: ");
sb.append(session.getid());
message = sb.tostring();
return success;
}
} 例6 classes/tutorial/iocservlet.java<%@ page contenttype="text/html; charset=utf-8" %>
<%@ taglib prefix="s" uri="/struts-tags"%>
<html>
<head>
<title>hello world!title>
head>
<body>
<h2>
<s:property value="message" escape="false"/>
<br>message from session: <s:property value="#session.msg"/>
h2>
body>
html>
例6 servlet.jsp<action name="noniocservlet" class="tutorial.noniocservlet">
<result>/servlet.jspresult>
action>
<action name="iocservlet" class="tutorial.iocservlet">
<result>/servlet.jspresult>
action>
例6 classes/struts.xml中noniocservlet和iocservlet action的配置运行tomcat,在浏览器地址栏中键入 或,出现如图7所示页面。
图7 servlet.jsp的输出页面
| 在servlet.jsp中,我用了两次property标志,第一次将escape设为false为了在jsp中输出 转行,第二次的value中的ognl为“#session.msg”,它的作用与session.getattribute("msg")等同。 关于property或其它标志,可以参考我的上一篇文章《常用的struts 2.0的标志(tag)介绍 》。 |
posted on 2006-10-25 12:10
max 阅读(99189)
评论(118) 所属分类:
struts 2.0系列