黑马程序员--Struts2框架技术

来源:互联网 发布:肉炖久了吃好吗 知乎 编辑:程序博客网 时间:2024/06/08 11:19

---------------------- android培训、java培训、期待与您交流! ------------------------------------------------------------------------------------------------------------------------------------------------

 

Struts2框架技术
Struts2是在webWork基础上发展而来的,和struts1一样,Structs也属于MVC框架,不过有一点大家需要注意的是:尽管Struts2和Struts1在名称上差别不是很大,但是两者的代码的编写风格几乎是不一样的。既然有了struts1,为什么还要有struts2呢?
因为Struts2存在以下优点。
1、 在软件设计上,Struts2没有想struts1那样跟ServletAPI和strutsAPI有着紧密的耦合。
Struts2的应用可以不依赖ServletAPI和strutsAPI.Struts2的这种设计属于无入侵式设计,而struts1却属于入侵式设计。
例如下边是struts1的一些代码:
public class OrderListAction extends Action {
 public ActionForward execute(ActionMapping mapping, ActionForm form,
   HttpServletRequest request, HttpServletResponse response)
   throws Exception {
 }
}
继承Action用到了strutsAPI,而HttpServletRequest,HttpServletRespons用到了servletAPI。
2、 Struts2提供了拦截器,利用拦截器可以进行AOP编程。实现权限拦截等功能。
3、 Struts2提供了类型转换器,我们可以吧特殊的请求参数转换为需要的类型,但是在struts1中我们要实现相同的功能,就必须向Struts1的底层实现BeanUtils注册类型转换器才行。
4、 Struts2提供支持多种表现层技术:如JSP,freeMarker,Velocity等。
5、 Struts2的输入校验可以对指定方法进行校验,而Struts1的校验功能是对action中的所有方法进行校验,这是不必要的。
6、 提供了全局范文(整个web应用),包范围(某个资源包中),和Action范围的国际化资源文件管理实现。
搭建Struts2开发环境:
1、 将Struts2所需要的jar文件添加到WEB-INF下的lib目录底下。
开发Struts 2程序最少需要的JAR。
struts2-core-2.x.x.jar :Struts 2框架的核心类库
xwork-core-2.x.x.jar :XWork类库,Struts 2在其上构建
ognl-2.6.x.jar :对象图导航语言(Object Graph Navigation Language),struts2框架通过其读写对象的属性
freemarker-2.3.x.jar :Struts 2的UI标签的模板使用FreeMarker编写
commons-logging-1.x.x.jar :ASF出品的日志包,Struts 2框架使用这个日志包来支持Log4J和JDK 1.4+的日志记录。
commons-fileupload-1.2.1.jar 文件上传组件,2.1.6版本后必须加入此文件
2、 编写Struts2的配置文件struts.xml
Struts2的默认配置文件是struts.xml.该文件存放在WEB-INF/classes下。所以在用eclipse开发的时候,放在src目录下。因为在发布到服务器的时候,服务器会将src下的非.java源文件的文件原照复制到相应的目录classes目录下。

我们可以从哪些地方获取到这些配置文件的模板呢?
可以在jar包中附带的例子中的WEB-INF/classes/的struts.xml文件中拷贝。

3、 在web.xml中加入Struts2MVC框架启动配置。

Struts2中,struts框架是通过Filter启动的,它在web.xml中的配置如下:
<filter>
    <filter-name>struts2</filter-name>
    <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
自从Struts 2.1.3以后,下面的FilterDispatcher已经标注为过时
<filter-class>org.apache.struts2.dispatcher.FilterDispatcher</filter-class>
</filter>
<filter-mapping>
    <filter-name>struts2</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

在StrutsPrepareAndExecuteFilter的init()方法中将会读取类路径下默认的配置文件struts.xml完成初始化操作。
注意:Struts2读取到struts.xml的内容后,以javabean的形式存放在内存中,以后struts2对用户的每次请求处理将使用内存中的数据。而不是每次都读取struts.xml文件。
这些可以从例子中的web.xml中获取。
对Struts2的MVC模式的图例:
控制器:Struts2框架中作为控制器的是StrutsPrepareAndExecuteFilter,他是一个servlet过滤器,当客户端请求时,首先要调用StrutsPrepareAndExecuteFilter过滤,由StrutsPrepareAndExecuteFilter决定该有哪个Action处理。
模型:Action在Struts2框架中时作为模型而存在的,主要包括两个功能,调用业务逻辑处理请求,和进行数据的传递,当Action对数据处理完以后,会返回一个逻辑视图。
视图:当视图组件接收到Action返回的逻辑视图后,会查找对应的物理视图资源,彬彬返回给客户端。


IE向某个JSP页面发出请求,JSP页面提交以.action为结尾的请求。

StrutsPrepareAndExecuteFilter接收请求并调用Action处理该请求。

Action处理完后统一返回一个逻辑视图。

StrutsPrepareAndExecuteFilter根据Action返回逻辑视图创建物理视图,将物理视图返回给页面。

注意在web应用中,Action看起来像是控制器,但是实际上它只是作为一个模型提供业务逻辑。真正的控制逻辑是StrutsPrepareAndExecuteFilter,由StrutsPrepareAndExecuteFilter调用Action中的控制逻辑。

 大致的过程:
浏览器请求某个资源,这时候会根据在web.xml中配置<filter-mapping>过滤所有的请求。
StrutsPrepareAndExecuteFilter是Struts 2框架的核心控制器,它负责拦截由
<url-pattern>/*</url-pattern>指定的所有用户请求,当用户请求到达时,该Filter会
过滤用户的请求。默认情况下,如果用户请求的路径不带后缀或者后缀以.action结尾,
这时请求将被转入Struts 2框架处理,否则Struts 2框架将略过该请求的处理。当请求
转入Struts 2框架处理时会先经过一系列的拦截器,然后再到Action。与Struts1不同,
Struts2对用户的每一次请求都会创建一个Action,所以Struts2中的Action是线程安全的

详细的过程:

浏览器请求某个资源,这时候会根据在web.xml中的配置

StrutsPrepareAndExecuteFilter接收到请求消息后,默认情况下会将后缀名为action的请求交给StrutsPrepareAndExecuteFilter类处理,并根据URLActionMapper中搜素指定Action的映射信息。

如果找到符合的映射信息,ActionProxy通过Configuration Manager在配置文件struts.xml中搜索被请求的Action类。

ActionProxy创建一个被请求的Action的实例,该实例将用来处理请求信息。

如果在struts.xml文件中存在与被请求Action相关的拦截器配置,那么该Action实例被调用的前后,这些拦截器也会被执行。

Action对请求处理完后,返回一个逻辑视图,由此逻辑视图寻找物理视图,并返回给客户端。

<package name="itcast"namespace="/test" extends="struts-default">

         <actionname="helloworld" class="cn.itcast.action.HelloWorldAction"method="execute" >

                   <resultname="success">/WEB-INF/page/hello.jsp</result>

         </action>

 </package>

struts2框架中使用包来管理Action,包的作用和java中的类包是非常类似的,它主要用于管理一组业务功能相关的action。在实际应用中,我们应该把一组业务功能相关的Action放在同一个包下。

 配置包时必须指定name属性,该name属性值可以任意取名,但必须唯一,他不对应java的类包,如果其他包要继承该包,必须通过该属性进行引用。包的namespace属性用于定义该包的命名空间,命名空间作为访问该包下Action的路径的一部分,如访问上面例子的Action,访问路径为:/test/helloworld.action namespace属性可以不配置,对本例而言,如果不指定该属性,默认的命名空间为“”(空字符串)。

通常每个包都应该继承struts-default包,因为Struts2很多核心的功能都是拦截器来实现。如:

从请求中把请求参数封装到action、文件上传和数据验证等等都是通过拦截器实现的。 struts-default定义了这些拦截器和Result类型。

可以这么说:当包继承了struts-default才能使用struts2提供的核心功能。 struts-default包是在struts2-core-2.x.x.jar文件中的META-INF中的struts-default.xml中定义。 struts-default.xml也是Struts2默认配置文件。 Struts2每次都会自动加载 struts-default.xml文件。

 包还可以通过abstract=true”定义为抽象包,抽象包中不能包含action

<action name="helloworld"class="cn.itcast.action.HelloWorldAction" method="execute" >

action的配置,name是给action起得一个名称,作为访问路径的一部分。

当请求到来时后,就会交给cn.itcast.action.HelloWorldAction类的Action实例的execute方法处理。

综上所述如果在浏览器中输入地址访问这个action的话,路径是这样的:

http://localhost:8080/web应用名称/包的命名空间/action的名称(可以加上.action后缀,也可以不加)

 

Action名称的搜索顺序:

1.获得请求路径的URI,例如url是:http://server/struts2/path1/path2/path3/test.action

 

2.首先寻找namespace/path1/path2/path3package,如果不存在这个package则执行步骤3;如果存在这个package,则在这个package中寻找名字为testaction,当在该package下寻找不到action时就会直接跑到默认namaspace(默认的命名空间为空字符串””)package里面去寻找action,如果在默认namaspacepackage里面还寻找不到该action,页面提示找不到action

 

3.寻找namespace/path1/path2package,如果不存在这个package,则转至步骤4;如果存在这个package,则在这个package中寻找名字为testaction,当在该package中寻找不到action时就会直接跑到默认namaspacepackage里面去找名字为testaction,在默认namaspacepackage里面还寻找不到该action,页面提示找不到action

 

4.寻找namespace/path1package,如果不存在这个package则执行步骤5;如果存在这个package,则在这个package中寻找名字为testaction,当在该package中寻找不到action时就会直接跑到默认namaspacepackage里面去找名字为testaction,在默认namaspacepackage里面还寻找不到该action,页面提示找不到action

 

5.寻找namespace/package,如果存在这个package,则在这个package中寻找名字为testaction,当在package中寻找不到action或者不存在这个package时,都会去默认namaspacepackage里面寻找action,如果还是找不到,页面提示找不到action

Action配置中的各项默认值

<package name="itcast"namespace="/test" extends="struts-default">

     <action name="helloworld"class="cn.itcast.action.HelloWorldAction" method="execute">

         <resultname="success">/WEB-INF/page/hello.jsp</result>

     </action>

 </package>

1>如果没有为action指定class,默认是ActionSupport

2>如果没有为action指定method,默认执行action中的execute()方法。

3>如果没有指定resultname属性,默认值为success

 

如果仅是为了实现:请求一个Action,转发的一个JSP页面的话,可以这样配置action

这样如果是访问这个addUI这个action的话,就会转发到employeeAdd.jsp这个页面。因为不配置class的话,class默认值为ActionSupportmethodexecute。而ActionSupportexecute方法返回的值为”success”.所以就会跳转到相应的页面。

Result的常用的视图类型

result配置在Struts2中提供了多种结果类型,常用的类型有:

dispatcher(默认值)如果不指定type属性的话,默认就是这种类型。

redirect:浏览器重定向都某个地址。

redirectAction:浏览器重定向到某个Action

plaintext:将请求的jsp等资源以源码的形式显示。多用于技术网站。

注意如果resulttyperedirect的话,那个重定向的jsp页面不能位于WEB-INF目录下边。

因为WEB-INF目录下的资源是受保护的,不能被浏览器直接访问。

 

result中还可以使用${属性名 }表达式访问action中的属性,表达式中的属性对应action中的属性。

多用到的场景是:当用户登录某个网站后,进入到某个可以修改信息的页面。当点击修改信息的按钮后,出现了修改信息的页面,但是用户乱填写,点击确定后,交给某个action来处理。action判断是错误的后,会用到重定向,还会回到当时填写信息的页面中,同时内容依旧是之前的内容,这时候就要从action中获取原来属性的信息。

redirectAction如果重定的action在同一个包下边的话。

 

<package name="itcast"namespace="/test" extends="struts-default">

 

<action name="helloworld"class="cn.itcast.action.HelloWorldAction" method="execute">

         <resultname="success">/WEB-INF/page/hello.jsp</result>

</action>

 

<action  name=”redirectionAction”>

   <result  type="redirectAction">helloworld</result>

</action>

 </package>

 

如果重定向的action在其他的命名空间下:

<result type="redirectAction">

        <param name="actionName">helloworld</param>

        <paramname="namespace">/test</param>

</result>

plaintext:显示原始文件内容,例如:当我们需要原样显示jsp文件源代码的时候,我们可以使用此类型。

<result name="source" type="plainText ">

        <paramname="location">/xxx.jsp</param>

        <paramname="charSet">UTF-8</param><!--指定读取文件的编码 -->

</result>

多个Action共享一个视图。全局result的配置:

当多个action中都使用到了相同的视图,这时,我们应该把result定义为全局的视图。

这时只能在这个包中可以用到的全局视图。

<package ....>

         <global-results>

                   <resultname="message">/message.jsp</result>

         </global-results>

</package>

 怎样能样整个web应用都能用到这个全局的视图呢?

可以是这样设计的:

先定义一个包:

 <packagename="base"  extends="struts-default">

      <global-results>

               <resultname="message">/WEB-INF/page/message.jsp</result>

      </global-results>

</package>

 这样其他的的包继承这个包:

<packagename="itcast"namespace="/test"extends="base"> 

<actionname="globalresult"class="cn.itcast.action.HelloWorldAction"method="add">

</action>

</package>

这样的话,如果在浏览器中范围itcast包中的这个globalresult这个action。就会用到全局的视图。因为base包继承了struts-default所以继承base包的所以包也就继承了struts-default

为Action的属性注入值:

Struts2Action中的属性提供了依赖注入的功能,在Struts2的配置文件中,可以很方便的为Action中的属性注入值。注意:如果注入的话,必须是属性具有setter方法。

public class HelloWorldAction{

         privateString savePath;

         publicString getSavePath() {

                   returnsavePath;

         }

         publicvoid setSavePath(String savePath) {

                   this.savePath= savePath;

         }

      ......

}

 <package name="itcast"namespace="/test" extends="struts-default">

         <actionname="helloworld" class="cn.itcast.action.HelloWorldAction">

                   <paramname="savePath">/images</param>

                   <resultname="success">/WEB-INF/page/hello.jsp</result>

         </action>

</package>

上面通过<param>节点为actionsavePath属性注入“/images

 

有些不适合写死的数据,通常用到的是Action的属性注入方式。

 

指定需要的Struts2处理的请求后缀

 前边都是使用.action后缀作为访问Action的后缀。其实,默认后缀是可以通过常量

“struts.action.extension”进行修改的。类如,我们可以配置Struts2只处理请求后缀为

.do的请求路径。

 <?xml version="1.0"encoding="UTF-8"?>

<!DOCTYPE struts PUBLIC

   "-//Apache Software Foundation//DTD Struts Configuration2.0//EN"

   "http://struts.apache.org/dtds/struts-2.0.dtd">

<struts>

   <constant name="struts.action.extension"value="do"/>

</struts>

 

如果用户需要指定多个请求后缀,则多个后缀之间以英文逗号(,)隔开。如:

 <constantname="struts.action.extension" value="do,go"/>

 

Struts2中常量的定义:

常量可以在struts.xmlstruts.properties中配置,建议在struts.xml中配置,两种配置方式如下:

struts.xml文件中配置常量

<struts>

   <constant name="struts.action.extension" value="do"/>

</struts>

 struts.properties中配置常量

struts.action.extension=do

 

 

    

先加载

  因为常量可以在下面多个配置文件中进行定义,所以我们需要了解struts2加载常量的搜索顺序:

struts-default.xml

struts-plugin.xml

 

  

 

 

struts.properties

web.xml

如果在多个文件中配置了同一个常量,则后一个文件中配置的常量值会覆盖前面文件中配置的常量值.最好是在struts.xml中定义。

Struts2中常用到的常量

指定默认编码集,作用于HttpServletRequestsetCharacterEncoding方法和freemarkervelocity的输出   

<constantname="struts.i18n.encoding" value="UTF-8"/>

该属性指定需要Struts 2处理的请求后缀,该属性的默认值是action,即所有匹配*.action的请求都由Struts2处理。

如果用户需要指定多个请求后缀,则多个后缀之间以英文逗号(,)隔开。

<constantname="struts.action.extension" value="do"/>

设置浏览器是否缓存静态内容,默认值为true(生产环境下使用),开发阶段最好关闭

<constantname="struts.serve.static.browserCache" value="false"/>

struts的配置文件修改后,系统是否自动重新加载该文件,默认值为false(生产环境下使用),开发阶段最好打开   

<constant name="struts.configuration.xml.reload"value="true"/>

开发模式下使用,这样可以打印出更详细的错误信息

<constantname="struts.devMode" value="true" />

默认的视图主题

<constantname="struts.ui.theme" value="simple" />

spring集成时,指定由spring负责action对象的创建

<constantname="struts.objectFactory" value="spring" />

该属性设置Struts 2是否支持动态方法调用,该属性的默认值是true。如果需要关闭动态方法调用,则可设置该属性为false

<constantname="struts.enable.DynamicMethodInvocation"value="false"/>

上传文件的总共大小限制

<constantname="struts.multipart.maxSize" value=“10701096"/>

在Struts2中为应用指定多个struts配置文件

 在大部分应用中,随着应用规模的增加,系统中的Action数量也会大量增加,导致struts.xml配置文件变得非常的臃肿,为了避免struts.xml文件过于庞大,过于臃肿。提高struts.xml文件的可读性。我们可以将一个struts.xml配置文件分解为多个配置文件,然后在struts.xml配置文件中包含其他的配置文件。

下边是struts.xml通过<include>元素指定多个配置文件。

<?xml version="1.0"encoding="UTF-8"?>

<!DOCTYPE struts PUBLIC

   "-//Apache Software Foundation//DTD Struts Configuration2.0//EN"

   "http://struts.apache.org/dtds/struts-2.0.dtd">

<struts>

         <includefile="struts-user.xml"/>

         <includefile="struts-order.xml"/>

</struts>

在实际生产中,配置文件要按照模块进行划分。最后在集成的时候,用<include>元素把各个模块的配置文件放置到struts.xml中。

动态方法的调用和使用通配符定义action

 

动态方法的调用:

如果Action中存在多个方法时,我们可以使用!+方法名调用指定方法。如下:

public class HelloWorldAction{

         privateString message;

         ....

         publicString execute() throws Exception{

                   this.message= "我的第一个struts2应用";

                   return"success";

         }

        publicString other() throws Exception{

                   this.message= "第二个方法";

                   return"success";

         }

}

假设访问上面actionURL路径为: /Struts2/test/helloworld.action

要访问actionother()方法,我们可以这样调用:

/Struts2/test/helloworld!other.action

如果不想使用动态方法调用,我们可以通过在struts.xml配置文件中配置常量struts.enable.DynamicMethodInvocation关闭动态方法调用。

<constantname="struts.enable.DynamicMethodInvocation"value="false"/>

 使用通配符定义action

<package name="itcast"namespace="/test" extends="struts-default">

         <actionname="helloworld_*" class="cn.itcast.action.HelloWorldAction"method="{1}">

                   <resultname="success">/WEB-INF/page/hello.jsp</result>

         </action>

</package>

public class HelloWorldAction{

         privateString message;

         ....

         publicString execute() throws Exception{

                   this.message= "我的第一个struts2应用";

                   return"success";

         }

       publicString other() throws Exception{

                   this.message= "第二个方法";

                   return"success";

         }

}

要访问other()方法,可以通过这样的URL访问:/test/helloworld_other.action

 通常通配符定义action比较灵活,不仅仅是上边这一种

Action接收请求参数:

Struts2内部的拦截器会将请求某个action的参数中与action具有属性名称相同的参数拦截。通过反射技术,获得Action的该属性的setter方法,从而用请求参数值,为action的该属性赋值。

采用基本类型接收请求参数(get/post)

Action类中定义与请求参数同名的属性,struts2便能自动接收请求参数并赋予给同名属性。

请求路径: http://localhost:8080/test/view.action?id=78

public class ProductAction {

    privateInteger id;

    publicvoid setId(Integer id) {//struts2通过反射技术调用与请求参数同名的属性的setter方法来获取请求参数值

        this.id = id;

    }

    publicInteger getId() {

return id;

}

}

采用复合类型接收请求参数

请求路径: http://localhost:8080/test/view.action?product.id=78

 public class ProductAction {

  private Product product;

  public void setProduct(Product product) {  this.product = product;  }

  public Product getProduct() {return product;}

}

Struts2首先通过反射技术调用Product的默认构造器创建product对象,然后再通过反射技术调用product中与请求参数同名的属性的setter方法来获取请求参数值。

 Struts2的自定义类型转换器:

有两种:

一种是局部的,只对某个action中的某个类型的类型转换起作用

一种是全局的,对所有的action中的该类型的类型转换起作用。

import java.util.Date;

public class HelloWorldAction {

         privateDate createtime;

         publicDate getCreatetime() {

                   returncreatetime;

         }

        publicvoid setCreatetime(Date createtime) {

                   this.createtime= createtime;

        }

}

java.util.Date类型的属性可以接收格式为2009-07-20的请求参数值。但如果我们需要接收格式为20091221的请求参数,我们必须定义类型转换器,否则struts2无法自动完成类型转换。

 public class DateConverter extendsDefaultTypeConverter {

                @Override  public Object convertValue(Map context,Object value, Class toType) {

         SimpleDateFormatdateFormat = new SimpleDateFormat("yyyyMMdd");

         try{

                   if(toType== Date.class){//当字符串向Date类型转换时

                            String[]params = (String[]) value;// Request.getParameterValues()

                            returndateFormat.parse(params[0]);

                   }elseif(toType == String.class){//Date转换成字符串时

                            Datedate = (Date) value;

                            returndateFormat.format(date);

                   }

         }catch (ParseException e) {}

         returnnull;

         }

}

将上面的类型转换器注册为局部类型转换器:

Action类所在的包下放置ActionClassName-conversion.properties文件,ActionClassNameAction的简单类名,后面的-conversion.properties是固定写法,对于本例而言,文件的名称应为HelloWorldAction-conversion.properties。在properties文件中的内容为:

属性名称=类型转换器的全类名

对于本例而言, HelloWorldAction-conversion.properties文件中的内容为:

createtime=cn.itcast.conversion.DateConverter

 此处定义的是双向的类型装换器:

如果是从客户端接收参数的话,那么是将20091221这种字符串类型的请求参数(value)转换为Date类型(toType)。

如果是将Date类型的数据回写到浏览器上,是将Date类型(value)的数据转换为字符串(toType)

 对该句代码的解释:

String[] params = (String[]) value;

为什么要转换为字符串数组。这是因为

例如:

复选框带过来的数据,名称一样,但是有好多个值。所以就要这样,实际内部调用的是

Request.getParameterValues()

 如果将上例注册为全局类型的转换器的话,步骤如下:

将上面的类型转换器注册为全局类型转换器:

WEB-INF/classes下放置xwork-conversion.properties文件。在properties文件中的内容为:

待转换的类型=类型转换器的全类名

对于本例而言, xwork-conversion.properties文件中的内容为:

java.util.Date=cn.itcast.conversion.DateConverter

在Action中访问或添加request、session、application属性

public String scope() throws Exception{

  ActionContext ctx = ActionContext.getContext();

  ctx.getApplication().put("app", "应用范围");//ServletContext里放入app

   ctx.getSession().put("ses","session范围");//session里放入ses

  ctx.put("req", "request范围");//request里放入req

  return "scope";

}

获取HttpServletRequest / HttpSession / ServletContext /HttpServletResponse对象

方式一:

 publicString rsa() throws Exception{

         HttpServletRequestrequest = ServletActionContext.getRequest();

         ServletContextservletContext = ServletActionContext.getServletContext();

         request.getSession()    

         HttpServletResponseresponse = ServletActionContext.getResponse();

         return"scope";

}

 文件上传:

如果要想用Struts2的文件的上传功能,则必须要使用的jar包是commons-fileupload-x.x.x.jar

同时最好是加入

commons-io-x.x.x.jar.主要是用到这个jar包中的一个工具类FileUtils.用来将上传的文件保存到服务器的某个目录底下。

第二步:把form表的enctype设置为:“multipart/form-data“,如下:

<formenctype="multipart/form-data" action="${pageContext.request.contextPath}/xxx.action"method="post">

 <input  type="file"name="uploadimage">

</form>

 第三步:在Action类中添加以下属性,属性红色部分对应于表单中文件字段的名称:

public class uploadAction{

 private File uploadImage;//得到上传的文件

 private String uploadImageContentType;//得到文件的类型

 private String uploadImageFileName;//得到文件的名称

  省略了gettersetter方法。

  publicString execute(){

        upload代码。

}

}

Action中属性的命名必须严格的按照Struts2的规范。

如果上传的文件超过了Struts2默认的最大值,可以通过常量的设置来改变上传文件大小的限制。<constant name="struts.multipart.maxSize"value=“10701096"/>

文件上传:

形如这样的:
<form enctype="multipart/form-data" action="${pageContext.request.contextPath}/xxx.action" method="post">
  <input  type="file" name="uploadImages">
  <input  type="file" name="uploadImages">
</form>
那么Action的属性设置为:
public class HelloWorldAction{
  private File[] uploadImages;//得到上传的文件
  private String[] uploadImagesContentType;//得到文件的类型
  private String[] uploadImagesFileName;//得到文件的名称
  //这里略省了属性的getter/setter方法

  public String execute(){
     upload方法。
    return “success”;
}
}
这样就可以做到多文件的上传。

Struts2的输入校验:

Struts2中,可以实现对action的所有的方法进行校验,也可以对action中指定的方法进行校验。

对于输入校验Struts2提供了两种实现方法:

1、采用手工编写代码实现。

2、基于XML配置方式的实现。

 方式1.

手工编写代码实现对Action中所有的方法进行输入校验

通过重写validate() 方法实现, validate()方法会校验action中所有与execute方法签名相同的方法。当某个数据校验失败时,我们应该调用addFieldError()方法往系统的fieldErrors添加校验失败信息(为了使用addFieldError()方法,action需要继承ActionSupport),如果系统的fieldErrors包含失败信息,struts2会将请求转发到名为inputresult。在input视图中可以通过<s:fielderror/>显示失败信息。

validate()使用例子:

public void validate() {

      if(this.mobile==null || "".equals(this.mobile.trim())){  this.addFieldError("username","手机号不能为空");

       }else{ if(!Pattern.compile("^1[358]\\d{9}").matcher(this.mobile.trim()).matches()){

                   this.addFieldError(“mobile","手机号的格式不正确"); }

      }

}

验证失败后,请求转发至input视图:

<resultname="input">/WEB-INF/page/addUser.jsp</result>

addUser.jsp页面中使用<s:fielderror/>显示失败信息。

方式2

手工编写代码实现对action指定方法输入校验。

通过validateXxx()方法实现,validateXxx()只会校验action中方法名为Xxx的方法。其中Xxx的第一个字母要大写。当某个数据校验失败时,我们应该调用addFieldError()方法往系统的fieldErrors添加校验失败信息(为了使用addFieldError()方法,action可以继承ActionSupport),如果系统的fieldErrors包含失败信息,struts2会将请求转发到名为inputresult。在input视图中可以通过<s:fielderror/>显示失败信息。

validateXxx()方法使用例子:

public String add() throws Exception{  return "success";}   

public void validateAdd(){

         if(username==null && "".equals(username.trim()))  this.addFieldError("username","用户名不能为空");

}

验证失败后,请求转发至input视图:

<resultname="input">/WEB-INF/page/addUser.jsp</result>

addUser.jsp页面中使用<s:fielderror/>显示失败信息。

输入校验的流程:

1。类型转换器对请求参数执行类型转换,并把转换后的值赋给action中的属性。

2。如果在执行类型转换的过程中出现异常,系统会将异常信息保存到ActionContextconversionError拦截器将异常信息添加到fieldErrors里。不管类型转换是否出现异常,都会进入第3步。

3。系统通过反射技术先调用action中的validateXxx()方法,Xxx为方法名。

4。再调用action中的validate()方法。

5。经过上面4步,如果系统中的fieldErrors存在错误信息(即存放错误信息的集合的size大于0),系统自动将请求转发至名称为input的视图。如果系统中的fieldErrors没有任何错误信息,系统将执行action中的处理方法。 

注意第3,4步骤我们是手工编写代码的形式将错误信息加入到了fieldErrors中。

综上,跳转到input视图的情况有3中。一种是类型转换有错误,一种是验证的时候出现错误。一种是类型转换和验证都有错误。

基于XMLAction中所有的方法进行输入校验

使用基于XML配置方式实现输入校验时,Action也需要继承ActionSupport,并且提供校验文件,校验文件和action类放在同一个包下,文件的取名格式为:ActionClassName-validation.xml,其中ActionClassNameaction的简单类名,-validation为固定写法。如果Action类为cn.itcast.UserAction,那么该文件的取名应为:UserAction-validation.xml。下面是校验文件的模版:

<?xml version="1.0"encoding="UTF-8"?>

<!DOCTYPE validators PUBLIC"-//OpenSymphony Group//XWork Validator 1.0.3//EN""http://www.opensymphony.com/xwork/xwork-validator-1.0.3.dtd">

<validators>

   <field name="username">

       <field-validator type="requiredstring">

           <param name="trim">true</param>

           <message>用户名不能为空!</message>

       </field-validator>

   </field>

</validators>

 

<field>指定action中要校验的属性,<field-validator>指定校验器,上面指定的校验器requiredstring是由系统提供的,系统提供了能满足大部分验证需求的校验器,这些校验器的定义可以在xwork-2.x.jar中的com.opensymphony.xwork2.validator.validators下的default.xml中找到。

<message>为校验失败后的提示信息,如果需要国际化,可以为message指定key属性,key的值为资源文件中的key

在这个校验文件中,对action中字符串类型的username属性进行验证,首先要求调用trim()方法去掉空格,然后判断用户名是否为空。

Struts2提供的校验器列表:

系统提供的校验器如下:

required (必填校验器,要求field的值不能为null)

requiredstring (必填字符串校验器,要求field的值不能为null,并且长度大于0,默认情况下会对字符串去前后空格)

stringlength(字符串长度校验器,要求field的值必须在指定的范围内,否则校验失败,minLength参数指定最小长度,maxLength参数指定最大长度,trim参数指定校验field之前是否去除字符串前后的空格)

regex(正则表达式校验器,检查被校验的field是否匹配一个正则表达式.expression参数指定正则表达式,caseSensitive参数指定进行正则表达式匹配时,是否区分大小写,默认值为true)

int(整数校验器,要求field的整数值必须在指定范围内,min指定最小值,max指定最大值)

double(双精度浮点数校验器,要求field的双精度浮点数必须在指定范围内,min指定最小值,max指定最大值)

fieldexpression(字段OGNL表达式校验器,要求field满足一个ognl表达式,expression参数指定ognl表达式,该逻辑表达式基于ValueStack进行求值,返回true时校验通过,否则不通过)

email(邮件地址校验器,要求如果field的值非空,则必须是合法的邮件地址)

url(网址校验器,要求如果field的值非空,则必须是合法的url地址)

date(日期校验器,要求field的日期值必须在指定范围内,min指定最小值,max指定最大值)

conversion(转换校验器,指定在类型转换失败时,提示的错误信息)

visitor(用于校验action中的复合属性,它指定一个校验文件用于校验复合属性中的属性)

expression(OGNL表达式校验器,expression参数指定ognl表达式,该逻辑表达式基于ValueStack进行求值,返回true时校验通过,否则不通过,该校验器不可用在字段校验器风格的配置中)

校验器的使用例子

required 必填校验器

<field-validatortype="required">

      <message>性别不能为空!</message>

</field-validator>

requiredstring 必填字符串校验器

<field-validatortype="requiredstring">

      <param name="trim">true</param>

      <message>用户名不能为空!</message>

</field-validator>

stringlength:字符串长度校验器

<field-validatortype="stringlength">

         <paramname="maxLength">10</param>

         <paramname="minLength">2</param>

         <paramname="trim">true</param>

         <message><![CDATA[产品名称应在2-10个字符之间]]></message>

</field-validator>

email:邮件地址校验器

<field-validatortype="email">

         <message>电子邮件地址无效</message>

</field-validator>

regex:正则表达式校验器

<field-validatortype="regex">

    <paramname="expression"><![CDATA[^1[358]\d{9}$]]></param>

    <message>手机号格式不正确!</message>

</field-validator>

int:整数校验器

<field-validatortype="int">

         <paramname="min">1</param>

         <paramname="max">150</param>

         <message>年龄必须在1-150之间</message>

</field-validator>

字段OGNL表达式校验器

<field name="imagefile">

         <field-validatortype="fieldexpression">

                   <paramname="expression"><![CDATA[imagefile.length() <=0]]></param>

                   <message>文件不能为空</message>

         </field-validator>

</field>

基于XML配置对Action中指定的方法进行输入校验:

当校验文件的取名为ActionClassName-validation.xml时,会对 action中的所有处理方法实施输入验证。如果你只需要对action中的某个action方法实施校验,那么,校验文件的取名应为:ActionClassName-ActionName-validation.xml,其中ActionNamestruts.xmlaction的名称。例如:在实际应用中,常有以下配置:

<action name="user_*"class="cn.itcast.action.UserAction" method="{1}“ >

         <resultname="success">/WEB-INF/page/message.jsp</result>

         <resultname="input">/WEB-INF/page/addUser.jsp</result>

</action>

UserAction中有以下两个处理方法:

public String add() throws Exception{

  ....

}

public String update() throws Exception{

  ....

}

要对add()方法实施验证,校验文件的取名为:   UserAction-user_add-validation.xml

要对update()方法实施验证,校验文件的取名为: UserAction-user_update-validation.xml  

基于XML校验的一些特点

当为某个action提供了ActionClassName-validation.xmlActionClassName-ActionName-validation.xml两种规则的校验文件时,系统按下面顺序寻找校验文件:

1AconClassName-validation.xml

2ActionClassName-ActionName-validation.xml

系统寻找到第一个校验文件时还会继续搜索后面的校验文件,当搜索到所有校验文件时,会把校验文件里的所有校验规则汇总,然后全部应用于action方法的校验。如果两个校验文件中指定的校验规则冲突,则只使用后面文件中的校验规则。

action继承了另一个action,父类action的校验文件会先被搜索到。

假设UserAction继承BaseAction

<action name="user"class="cn.itcast.action.UserAction" method="{1}">

</action>

访问上面action,系统先搜索父类的校验文件:BaseAction-validation.xmlBaseAction-user-validation.xml,接着搜索子类的校验文件: UserAction-validation.xmlUserAction-user-validation.xml。应用于上面action的校验规则为这四个文件的总和

 

Struts2的国际化:

准备资源文件,资源文件的命名格式如下:

baseName_language_country.properties

baseName_language.properties

baseName.properties

其中baseName是资源文件的基本名,我们可以自定义,但languagecountry必须是java支持的语言和国家。如:

中国大陆: baseName_zh_CN.properties

美国: baseName_en_US.properties

 

现在为应用添加两个资源文件:

第一个存放中文:itcast_zh_CN.properties

内容为:welcome=欢迎来到中国

第二个存放英语(美国): itcast_en_US.properties

内容为: welcome=welcome to China

 

国际化输出带占位符的国际化信息

资源文件中的内容如下:

welcome= {0},欢迎来到传智播客{1}

jsp页面中输出带占位符的国际化信息

<s:text name="welcome">

       <s:param>黎明</s:param>

         <s:param>学习</s:param>

</s:text>

Action类中获取带占位符的国际化信息,可以使用getText(String key, String[] args)getText(StringaTextName, List args)方法。

 

配置全局资源与输出国际化信息

当准备好资源文件之后,我们可以在struts.xml中通过struts.custom.i18n.resources常量把资源文件定义为全局资源文件,如下:

<constantname="struts.custom.i18n.resources" value="itcast" />

itcast为资源文件的基本名。

后面我们就可以在页面或在action中访问国际化信息:

JSP页面中使用<s:text name=“”/>标签输出国际化信息:

<s:text  name=“user”/>name为资源文件中的key

因为全局的资源配置的话,整个web应用都可以获取到国际化资源信息,所以这个无关于Action

Action类中,可以继承ActionSupport,使用getText()方法得到国际化信息,该方法的第一个参数用于指定资源文件中的key

国际化包范围资源文件

在一个大型应用中,整个应用有大量的内容需要实现国际化,如果我们把国际化的内容都放置在全局资源属性文件中,显然会导致资源文件变的过于庞大、臃肿,不便于维护,这个时候我们可以针对不同模块,使用包范围来组织国际化文件。

方法如下:

java的包下放置package_language_country.properties资源文件,package为固定写法,处于该包及子包下的action都可以访问该资源。当查找指定key的消息时,系统会先从package资源文件查找,当找不到对应的key时,才会从常量struts.custom.i18n.resources指定的资源文件(全局配置)中寻找。

国际化—Action范围资源文件

我们也可以为某个action单独指定资源文件,方法如下:

Action类所在的路径,放置ActionClassName_language_country.properties资源文件,ActionClassNameaction类的简单名称。

OGNL表达式:

OGNLObject GraphicNavigation Language(对象图导航语言)的缩写,它是一个开源项目。 Struts 2框架使用OGNL作为默认的表达式语言。

相对EL表达式,它提供了平时我们需要的一些功能,如:

支持对象方法调用,如xxx.sayHello()

支持类静态方法调用和值访问,表达式的格式为@[类全名(包括包路径)]@[方法名值名],例如:@java.lang.String@format('foo%s', 'bar')@cn.itcast.Constant@APP_NAME

操作集合对象。

Ognl 有一个上下文(Context)概念,说白了上下文就是一个MAP结构,它实现了java.utils.Map接口,在Struts2中上下文(Context)的实现为ActionContext,下面是上下文(Context)的结构示意图

Struts2接受一个请求时,会迅速创建ActionContextValueStackaction。然后把action存放进ValueStack,所以action的实例变量可以被OGNL访问。

访问上下文(Context)中的对象需要使用#符号标注命名空间,如#application#session

另外OGNL会设定一个根对象(root对象),在Struts2中根对象就是ValueStack(值栈)。如果要访问根对象(即ValueStack)中对象的属性,则可以省略#命名空间,直接访问该对象的属性即可。

struts2中,根对象ValueStack的实现类为OgnlValueStack,该对象不是我们想像的只存放单个值,而是存放一组对象。在OgnlValueStack类里有一个List类型的root变量,就是使用他存放一组对象

               |--request  

               |--application  

context ------|--OgnlValueStack root变量[action,OgnlUtil, ... ]  

               |--session  

               |--attr  

               |--parameters

root变量中处于第一位的对象叫栈顶对象。通常我们在OGNL表达式里直接写上属性的名称即可访问root变量里对象的属性,搜索顺序是从栈顶对象开始寻找,如果栈顶对象不存在该属性,就会从第二个对象寻找,如果没有找到就从第三个对象寻找,依次往下访问,直到找到为止。

大家注意:Struts2中,OGNL表达式需要配合Struts标签才可以使用。如:

<s:property value="name"/>

此处的的name就是一个ognl表达式,因为没有写#符号,所以会到值栈中去找该属性,

通常值栈中第一个对象为action对象,所以就会去action对象中找是否是name这个属性。

 

由于ValueStack(值栈)Struts 2OGNL的根对象,如果用户需要访问值栈中的对象,在JSP页面可以直接通过下面的EL表达式访问ValueStack(值栈)中对象的属性:

${foo} //获得值栈中某个对象的foo属性.

 

如果访问其他Context中的对象,由于他们不是根对象,所以在访问时,需要添加#前缀,配合Struts2的标签使用。

l  application对象:用于访问ServletContext,例如#application.userName或者#application['userName'],相当于调用ServletContextgetAttribute("username")

l  session对象:用来访问HttpSession,例如#session.userName或者#session['userName'],相当于调用session.getAttribute("userName")

l  request对象:用来访问HttpServletRequest属性(attribute)的Map,例如#request.userName或者#request['userName'],相当于调用request.getAttribute("userName")

l  parameters对象:用于访问HTTP的请求参数,例如#parameters.userName或者#parameters['userName'],相当于调用request.getParameter("username")

l  attr对象:用于按page->request->session->application顺序访问其属性。

为什么用EL表达式可以直接访问ValueStack中对象的属性呢?

EL表达式以前说的只能访问:page/request/session/application.

EL表达式底层调用的方法是:pageContexst.findAttribute(“属性值”)。找寻属性的时候,还是调用的 域对象.getAttribute(“属性”)

 

原因是Struts2HttpServletRequest作了进一步的封装。覆写了requestgetAttribute()方法。简略代码如下:

 public class StrutsRequestWrapper extendsHttpServletRequestWrapper {

      public StrutsRequestWrapper(HttpServletRequest req) {

          super(req);

      }

      public Object getAttribute(String   key) {

       ......

      ActionContext ctx = ActionContext.getContext();

      Object attribute = super.getAttribute(key);//先从request范围获取属性值

   if(ctx != null) {

      if (attribute == null) {//如果从request范围没有找到属性值,即从ValueStack中查找对象的属性值

               ......

      ValueStack stack = ctx.getValueStack();

      attribute = stack.findValue(key);

               ......

           }

       }

       return attribute;

    }

 }

采用OGNL表达式创建List/Map集合对象

 

 如果需要一个集合元素的时候(例如List对象或者Map对象),可以使用OGNL中同集合相关的表达式。
<s:set>标签的特点:
Set标签用于将某个值放入指定范围。 Set标签有个scope属性,用于指定放置的范围。
scope:指定变量被放置的范围,该属性可以接受application、session、request、 page或action。如果没有设置该属性,则默认放置在OGNL Context中
<s:iterator>标签的特点:会将从集合中迭代出来的对象放置到OGNL Context的root变量的栈顶。
例如list集合中存放的是{new Book(),new Person(),new Food()}.使用了iterator标签后,root变量中的对象为root{new Book(),new Person(),new Food(),action,……….}.
如果是Map集合,那么root{entry,action}。其中entry是Map.entry<key,value>.
<s:property/>标签如果不指定value值的话,默认从root变量的栈顶去将对象取出。
如果指定value的话,从栈顶中的第一个对象开始查找value属性。

使用如下代码直接生成一个List对象:
如果不指定scope属性的话,默认会在OGNL Context中,和Application,session是一个级别的。
 <s:set name="list" value="{'zhangming','xiaoi','liming'}" />
迭代list中的对象的属性。因为list位于OGNL Context中,所以要加上#标志。
<s:iterator value="#list" >会将”zhangming”,”xiaoi”,”liming”三个字符串对象放入值栈的栈顶。
 <s:property/><br>//从栈顶取出”zhangming”这第一个字符串对象。
</s:iterator>
生成一个Map对象:
<s:set name="foobar" value="#{'foo1':'bar1', 'foo2':'bar2'}" />
<s:iterator value="#foobar" >将各个entry对象依次放入栈顶。
 <s:property value="key"/>=<s:property value="value"/><br>  从栈顶entry中查找属性为key和value的属性值。
</s:iterator>
value:赋给变量的值.如果没有设置该属性,则将ValueStack栈顶的值赋给变量。

采用OGNL表达式判断对象是否存在于集合中

对于集合类型,OGNL表达式可以使用innot in两个元素符号。其中,in表达式用来判断某个元素是否在指定的集合对象中;not in判断某个元素是否不在指定的集合对象中,如下所示。

in表达式:

<s:if test="'foo' in{'foo','bar'}">

  

</s:if>

<s:else>

   不在

</s:else>

not in表达式:

<s:if test="'foo' not in{'foo','bar'}">

   不在

</s:if>

<s:else>

  

</s:else>

OGNL表达式的投影功能

除了innot in之外,OGNL还允许使用某个规则获得集合对象的子集,常用的有以下3个相关操作符。

?:获得所有符合逻辑的元素。

^:获得符合逻辑的第一个元素。

$:获得符合逻辑的最后一个元素。

例如代码:

<s:iteratorvalue="books.{?#this.price > 35}">

     <s:property value="title" /> - $<s:propertyvalue="price" /><br>

</s:iterator>

在上面代码中,直接在集合后紧跟.{}运算符表明用于取出该集合的子集,{}内的表达式用于获取符合条件的元素,this指的是为了从大集合books筛选数据到小集合,需要对大集合books进行迭代,this代表当前迭代的元素。本例的表达式用于获取集合中价格大于35的书集合。

public class BookAction extendsActionSupport {

         privateList<Book> books;

         ....

         @Override

             public String execute() {

                          books = new LinkedList<Book>();

                          books.add(newBook("A735619678", "spring", 67));

                   books.add(newBook("B435555322", "ejb3.0",15));

         }

}

Struts2的常用标签:

property标签

property标签用于输出指定值:

<s:set name="name"value="'kk'" />

<s:property value="#name"/>

default:可选属性,如果需要输出的属性值为null,则显示该属性指定的值

escape:可选属性,指定是否格式化HTML代码。

value:可选属性,指定需要输出的属性值,如果没有指定该属性,则默认输出ValueStack栈顶的值。

id:可选属性,指定该元素的标识

iterator标签

iterator标签用于对集合进行迭代,这里的集合包含ListSet和数组。

<s:set name="list"value="{'zhangming','xiaoi','liming'}" />

<s:iterator value="#list"status="st">

         <fontcolor=<s:iftest="#st.odd">red</s:if><s:else>blue</s:else>>

         <s:property/></font><br>

</s:iterator>

value:可选属性,指定被迭代的集合,如果没有设置该属性,则使用ValueStack栈顶的集合。

id:可选属性,指定集合里元素的id

status:可选属性,该属性指定迭代时的IteratorStatus实例。该实例包含如下几个方法:

         intgetCount(),返回当前迭代了几个元素。

         intgetIndex(),返回当前迭代元素的索引。

         booleanisEven(),返回当前被迭代元素的索引是否是偶数

         booleanisOdd(),返回当前被迭代元素的索引是否是奇数

         booleanisFirst(),返回当前被迭代元素是否是第一个元素。

         booleanisLast(),返回当前被迭代元素是否是最后一个元素。

if/elseif/else标签

<s:set name="age"value="21" />

<s:if test="#age==23">

         23

</s:if>

<s:elseif test="#age==21">

         21

</s:elseif>

<s:else>

         都不等

</s:else>

url标签

<s:url action="helloworld_add"namespace="/test"><s:param name="personid"value="23"/></s:url>

生成类似如下路径:

/struts/test/helloworld_add.action?personid=23

红色部分为内容路径。

当标签的属性值作为字符串类型处理时,“%”符号的用途是计算OGNL表达式的值。

 <s:set name="myurl"value="'http://www.foshanshop.net'"/>

  <s:url value="#myurl" /><br>

  <s:url value="%{#myurl}" />

输出结果:

#myurl

http://www.foshanshop.net

<s:token />标签防止重复提交

<s:token />标签防止重复提交,用法如下:

第一步:在表单中加入<s:token />

<s:formaction="helloworld_other" method="post"namespace="/test">

 <s:textfieldname="person.name"/><s:token/><s:submit/>

 </s:form>

第二步:

<action name="helloworld_*"class="cn.itcast.action.HelloWorldAction" method="{1}">

      <interceptor-ref name="defaultStack" />

      <interceptor-ref name="token" />

      <resultname="invalid.token">/WEB-INF/page/message.jsp</result> 

      <result>/WEB-INF/page/result.jsp</result>             

</action>

以上配置加入了“token”拦截器和“invalid.token”结果,因为“token”拦截器在会话的token与请求的token不一致时,将会直接返回“invalid.token”结果。

debug状态,控制台出现下面信息,是因为Action中并没有struts.tokenstruts.token.name属性,我们不用关心这个错误:

严重: ParametersInterceptor - [setParameters]: Unexpected Exceptioncaught setting 'struts.token' on 'class xxx: Error setting expression'struts.token' with value '[Ljava.lang.String;@39f16f'

严重: ParametersInterceptor - [setParameters]: Unexpected Exceptioncaught setting 'struts.token.name----------------------android培训、java培训、期待与您交流! --------------------------------------------------------------------------------------------------------------------------------------------------

详细请查看:http://edu.csdn.net/heima

原创粉丝点击