Struts2入门2 Struts2深入

时间:2023-03-08 21:52:16

Struts2入门2 Struts2深入

链接: http://pan.baidu.com/s/1rdCDh 密码: sm5h

前言:

前面学习那一节,搞得我是在是太痛苦了。因为在Web项目中确实不知道该添加那些Struts2的包,不该添加那些。当我一股脑的将所有的包都加进去的时候,悲剧照样是发生,估计是因为Web项目导致的。将上面那个例子配置好真心费事,然后就是最好不要使用FilterDispatcher这个类当做Struts2在web.xml配置的拦截器,转到使用org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter.

这一节我们深入的学习一下Struts2的知识:struts.xml文件的主要元素、Action的实现方式、Action对于ActionContext的访问方式、Action对于ServletAPI的访问方式、Action的配置方式以及通配符的使用、掌握处理结果的流程、利用通配符配置result、掌握dispatcher,redirect,redirectAction等结果类型的使用、掌握Struts异常机制。

1.配置文件详解

Struts2配置文件是整个框架的联系纽带,Struts通过配置文件将核心控制器StrutsPrepareAndExecuteFilter、业务控制器Action和视图组件联系在一起。配置文件在Struts2框架中起着调度的作用,同时配置文件的使用也降低了类之间的耦合程度。Struts2的核心配置文件就是struts.xml,负责管理Struts2框架的业务控制器Action。默认情况下放在类的根目录下面。

1.1常量的配置

在struts.properties、struts.xml或者是web.xml中配置。常量的加载顺序是struts.xml,struts.properties, web.xml,所以如果在struts.xml中定义的A,如果在web.xml中也定义了,就会出现web.xml覆盖struts.xml中的A

常见的属性:

struts.configuration

struts.local

struts.i18n.encoding

struts.xslt.nocache

struts.configuration.files 加载struts2的多个配置文件,多个文件使用,隔开

struts.multipart.saveDir 长传文件的临时保存目录

struts.multipart.maxSize 文件最大的字节数

struts.customer.propertiex 指定加载自定义的属性文件

struts.mapper.class         指定HTTP请求映射到Action的映射器,默认是org.apache.struts2.mapper.DefaultActionMapper

struts.action.extension 指定请求后缀,默认是action,也就是所有的后缀是.action的请求都是有Struts2处理的

struts.serve.static.browserCache

struts.configuration.xml.reload      struts.xml修改之后是否重新加载该文件

struts.devMode 开发模式

1.2包配置

Struts2中包的概念,每一个包就是有若干个Action、拦截器和拦截器引用组成的集合。Struts2通过包来管理Action和拦截器等核心组件。

<package name=”packagename” extends=”struts-default” namespace=”” abstract=””>

Abstract抽象包是不可以定义<action>

1.3命名空间配置

通过创建包,可以将逻辑上相关的Action或者是拦截器等组件放在同一个包中,

<package name=”p1” extends=”struts-default”>

<action name=”login” class=”com.yang.stru2.action.LoginAction”>

<result name=”error”>/error.jsp</result>

<result name=”success”>/success.jsp</result>

</action>

</package>

<package name=”p2” extends=”struts-default”>

<action name=”login” class=”com.yang.stru2.action.LoginAction”>

<result name=”error”>/error.jsp</result>

<result name=”success”>/success.jsp</result>

</action>

</package>

这种情况下,p1和p2包中都定义了一个login的action,当请求访问p1的Action就会发现负责处理请求的始终是p2中的Action,因为p2将p1中的请求给覆盖掉了,使得p1中的Action没有机会处理请求,可以使用namespace来解决。

<package name=”p1” extends=”struts-default” namespace=”/admin”>

<action name=”login” class=”com.yang.stru2.action.LoginAction”>

<result name=”error”>/error.jsp</result>

<result name=”success”>/success.jsp</result>

</action>

</package>

<package name=”p2” extends=”struts-default”>

<action name=”login” class=”com.yang.stru2.action.LoginAction”>

<result name=”error”>/error.jsp</result>

<result name=”success”>/success.jsp</result>

</action>

</package>

这样,当用户请求是namespace+action的规则: http://ip:port/WebProj/admin/login.action

访问p2中的请求 http://ip:port/webProj/login.action

如果不指定命名空间的话,默认是namespace=””.某个包可以显示的指定命名空间是根目录namespace=”/”

Struts2执行Action搜索:

查找指定命名空间下的Action,找到则执行;如果找不到,则转入到默认的命名空间下查找Action,找到则执行;还没有找到的话,就会报错。

1.4包含其他文件的配置信息

使用<include file=”bookshop-product.xml”/>位于struts根目录下,通常情况下,会将所有的struts的配置文件放到根目录下,其中饮用的每一个配置文件必须是有效的struts的配置文件

2.Action详解

对于Struts2 来说Action是应用的核心,每一个Action类就是一个工作单元,包含了对于用户请求的逻辑处理,因此Action也被称为业务控制器。在开发过程中,开发者需要根据处理逻辑的不同写出不同的Action诶,并且在struts.xml中配置好每一个Action类。Struts2框架负责将请求和Action进行匹配,匹配成功的话,这个Action就会被Struts2框架调用,进而完成请求的处理。

2.1Action的实现

POJO方式实现,必须包含一个特定的方法public String execute(),在上面的例子中已经实现过了,这里注意参数传递的时候,Action中的属性名字不一定要和表单中的元素名称完全一致,但是对已表单中的每一个元素名在Action中一定要有对应的getter/setter方法。比如表单中的是userName,那么在Action中必须有setUserName() & getUserName(),但是Action的对应的属性名臣却可以是不同的。

实现Action接口,统一execute返回的规范:

继承ActionSupport类:提供了Action的一个实现类就是ActionSupport,该类提供了许多方法,比如国际化信息的方法,数据校验,默认处理用户请求的方法,实际上ActionSupport是Struts2默认的Action的处理类,当用户在Action没有指定Action的时候使用的是这个ActionSupport

package com.yang.stru2.action;

import com.opensymphony.xwork2.ActionSupport;

public class RegAction extends ActionSupport {

private String userName;

private String password;

public String getUserName(){ return this.userName;}

public void setUserName(String userName){this.userName = userName;}

public String getPassword(){return this.password;}

public void setPassword(String password){ this.password = password;}

@Override

public void validate(){

if(this.userName == null ||this.userName.equals( "")){

this.addFieldError("userName", "用户名不可以为空");

System.out.print("用户名为空");

}

if(this.password== null || this.password.equals("")){

this.addFieldError("password", "密码不可以为空");

System.out.println("密码为空");

}

}

@Override

public String execute(){

System.out.println("---------注册用户信息如下------");

System.out.println("用户名:" + userName);

System.out.println("密码:" + password);

return SUCCESS;

}

}

<package name="reg" extends="struts-default">

<action name="reg" class="com.yang.stru2.action.RegAction">

<result name="success">/reg_success.jsp</result>

<result name="input">/reg.jsp</result>

</action>

</package>

这个比较智能,所以推荐使用ActionSupport这种方式,在执行execute之前还可以进行数据校验。

2.2Action访问ActionContext

Struts2没有直接和ServletAPI进行耦合,这是Struts2的改进,但是Action也需要访问ServletAPI,比如HttpSession HttpServletSession ServletContext

在Struts2中提供了一个ActionContext的class,Struct可以通过它来访问ServletAPI:

Object get(Object key)

Map getApplication() 返回一个Map对象,类似ServtletContext

ActionContext getContext() 获取系统的ActionContext

Map getParameters()

Map getSession()

void setApplication(Map application)

void setSession(Map session);

ActionContext是Action的执行上下文,Action在执行的时候需要用到相关的Servlet对象比如request, session,所以每一次执行的时候都会创建一个ActionContext对象的副本,多线程下访问是不会出现问题的(其实在Struts2内部使用的是ThreadLocal对象来处理ActionContext)

访问Servlet资源

public class CounterAction extends ActionSupport {

@Override

public String  execute(){

ActionContext actionContext = ActionContext.getContext();

Integer counter = (Integer) actionContext.getApplication().get("counter");

if(counter == null){

counter = 1;

System.out.println("counter null");

}else{

counter++;

}

actionContext.getApplication().put("counter", counter);

System.out.println("counter:"+counter);

return SUCCESS;

}

}

2.3Action直接访问ServletAPI

虽然可以通过ActionContext间接地操作ServletAPI,但是有的时候需要在Action中直接访问Servlet API.Struts2 提供了一系列的接口,通过Action实现这些接口便可以直接访问Servlet API:

ServletContextAware 对应 ServletContext

ServletRequestAware 对应HttpServletRequest

ServletResponseAware 对应 HttpServletResponse

当然了在Action中还可以使用ServletActionContext类来间接操作Servlet API:

static PageContext getPageContext()

static HttpServletRequest getRequest()

static HttpServletResponse getResponse()

static ServletContext getServletContext()

在开发过程只能够,为了避免Action于Servlet API直接耦合,推荐使用ActionContext来间接操作ServletAPI,如果在Action中必须使用Servlet API,可以通过ServletActionContext类的帮助,从而以更加简单的方式直接访问Servlet API。

2.4Action的配置

编写Action之后,需要在struts.xml文件中配置对应的Action信息,从而使Struts2框架知道该Action的存在,从而调用该Action来处理用户请求。

Struts2 使用包的概念组织action,在struts.xml中通过使用关键字package下的action元素配置Action。在配置Action的时候,需要指定action的name属性,action还有一个class属性指定器实现的类,但这个class属性不是必须的,默认使用的是ActionSupport 类

2.5动态方法的调用

在实际的开发中通常一个action对应多个处理逻辑,比如一条记录有编辑和删除,但是分别交给Action的不同方法处理,可以根据actionName!methodName.action的形式访问Action,比如reg!del.action直接访问RegAction中的del()方法。这种方式被称为DMI(Dynamic Method Invocation,动态方法调用)

<a href=”user!del.action”></a>

<a href=”user!edit.action”></a>

在UserAction.java 中增加两个方法,返回值是String

配置信息struts.xml

<package name=”user” extends=”struts-dedault”>

<action=”user” class=”com.yang.stru2.action.UserAction”>

<result name=”del”>/user/del.jsp</result>

</action>

</package>

同时需要在struts.properties中配置DMI开启

struts.enable.DynamicMethodInvocation=true,否则默认是会将它关闭的

2.6通配符匹配方式

除了DMI方式之外,还可以使用配置通配符来使用Action不同方法处理请求,在配置action元素的时候,需要指定name, class, method属性,这三个属性都支持通配符,但是不建议使用

YangTengfei

2013.11.30