zoukankan      html  css  js  c++  java
  • Struts2 笔记

    Struts2是在WebWork2基础发展而来的。和Struts1一样,Struts2也属于MVC框架。尽管二者在名字上的差别不是很大,但是在代码编写风格上几乎是不一样的。

    Struts2主要有以下优点:

    • 在软件设计上Struts2没有像Struts1那样更ServletAPI有着紧密的耦合,Struts2的应用可以不依赖于ServletAPI和StrutsAPI。Struts2的这种设计属于无侵入式设计,而Struts1却属于侵入式设计。
    • Struts2提供了拦截器,利用拦截器可以进行AOP编程,实现如权限拦截等功能。
    • Struts2提供了类型转换器,可以把特殊的请求参数转换成需要的类型。在Struts1中,如果要实现同样的功能,就必须向Struts1的底层实现BeanUtil注册类型转换器才行。
    • Struts2提供支持多种表现层技术。如:JSP、freeMarker、Velocity等。
    • Struts2的输入校验可以对指定方法进行校验,解决了Struts1长久之痛。
    • 提供了全局范围、包范围和Action范围的国际化资源文件管理实现。

    1. 环境搭建:

      搭建Struts2环境,需要以下几个步骤:

    • 找到开发Struts2应用需要使用到的jar文件,在发行包的lib目录中(不同版本需要的最小jar包是不同的,参见不同版本的文档 2.1.7)

        struts2-core.jar    Struts2框架的核心类库
        xwork-2.jar    xwork类库,Struts2在其上构建
        ognl.jar    对象图导航语言(Object Graph Navigation Language),Struts2框架通过其读取对象的属性
        freemarker.jar    Struts2UI标签的模板使用freeMarker编写
        commons-logging.jar    ASF出品的日志包,Struts2框架使用这个日志包来支持Log4J和JDK1.4+的日志记录
        commons-fileupload.jar    文件上传组件,2.1.6版本后必须加入此文件
        commons-io.jar    文件上传依赖的包

    • 编写Struts2的配置文件

        Struts2默认的配置文件为struts.xml,该文件需要存放在WEB-INF/classes下,该文件的配置模板如下:

    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE struts PUBLIC
        "-//Apache Software Foundation//DTD Struts Configuration 2.1.7//EN"
        "http://struts.apache.org/dtds/struts-2.1.7.dtd">
    <struts>
    </struts>
    • 在web.xml中加入Struts2MVC框架启动配置

        在Struts1.x中,Struts框架是通过Servlet启动的。在Struts2,Struts框架是通过Filter启动的。在web.xml中的配置如下

      <filter>
          <filter-name>struts2</filter-name>
          <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
      </filter>
      <filter-mapping>
          <filter-name>struts2</filter-name>
          <url-pattern>/*</url-pattern>
      </filter-mapping>

    2. 第一个Struts2应用--HelloWorld

     (1)编写struts.xml配置文件

    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE struts PUBLIC
        "-//Apache Software Foundation//DTD Struts Configuration 2.1.7//EN"
        "http://struts.apache.org/dtds/struts-2.1.7.dtd">
    <struts>
        <!-- 
        package:方便管理动作元素
            name:必须有。包的名称,配置文件中必须保证唯一
            namespace:该包的名称空间,一般是以“/”开头
            extends:继承的父包的名称(在struts2-core.jar中有一个struts-default.xml)
            abstract:是否是抽象包。没有任何action元素就是抽象包
        
        action:代表一个请求动作
            name:同包中必须唯一。动作的名称
            class:负责处理的JavaBean的类全名
            method:JavaBean中的对应处理方法。动作方法,特点是:public String 方法名(){}
        
        result:结果类型
            name:动作方法返回的字符串
            主体内容:View的具体地址
         -->
        <package name="ztq" namespace="/test" extends="struts-default">
            <action name="helloworld" class="me.ztq.action.HelloWorldAction" method="sayHello">
                <result name="success">/1.jsp</result>
            </action>
        </package>
    </struts>

    struts.xml配置中的包介绍:

    在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文件中的struts-default.xml中定义。struts-default.xml也是Struts2默认配置文件。Struts2每次都会自动加载struts-default.xml文件。

    (2)根据配置文件,创建需要的JavaBean和对应的动作方法,在动作方法中完成逻辑调用。

     1 package me.ztq.action;
     2 
     3 import java.io.Serializable;
     4 
     5 public class HelloWorldAction implements Serializable {
     6     private String message;
     7 
     8     public String getMessage() {
     9         return message;
    10     }
    11 
    12     public void setMessage(String message) {
    13         this.message = message;
    14     }
    15     
    16     public String sayHello(){
    17         message = "helloworld by struts2";
    18         return "success";
    19     }
    20 }

    (3)编写View,显示结果。

      ${message}

    (4)访问helloworld动作的方式:http://localhost:8080/struts2text/test/helloworld

      默认情况下,访问动作名helloworld,可以直接写helloworld也可以helloworld.action

    3. struts配置文件详解

     (1)action:

        class:默认值是com.opensymphony.xwork2.ActionSupport

          常量:SUCCESS   success 

             NONE  none

             ERROR  error

             INPUT  input

             LOGIN  login

         method:默认值是public String execute(){}

        实际开发中,自己编写的动作类一般情况下继承ActionSupport类。

     (2)result:

        type:转到目的地的方式。默认值是转发,名称是dispatcher

        (注:type的取值是定义好的,不是随便写的。在struts-default.xml中的package中有定义)

          dispatcher:普通的转发到某个页面   

          redirect:重定向到一个页面

          chain:普通的转发到某个动作名称

          redirectAction:重定向到一个动作名称

          plainText:以纯文本的形式输出JSP内容

      result元素的写法:

      (一)<result type="chain" name="success">a2</result>

      (二)<result type="chain" name="success">

           <param name="actionName">a2</param> <!--name对应的是chain处理器中的setActionName()方法-->

         </result>

      (注:如果要转向的是在另外一个名称空间的动作,那么只能用(二))

      如:

        <package name="p1" namespace="/namespace1" extends="struts-default">
            <action name="a2">
                <result type="dispatcher" name="success">/3.jsp</result>
            </action>
        </package>
        <package name="p2" namespace="/namespace2" extends="struts-default">
            <action name="a1">
                <result type="chain" name="success">
                    <param name="namespace">/namespace1</param>
                    <param name="actionName">a2</param>
                </result>
            </action>
        </package>

      访问:http://localhost:8080/struts2text/namespace2/a1

     (3)

      开发中配置文件的更改,在访问时让框架自动重新加载:

      struts.devMode = false(在default.properties中)

      利用struts.xml中的constant元素来覆盖掉default.properties中的默认行为

      <struts>

        <constant name="struts.devMode" value="true"></constant>

      </struts>

      指定需要Struts2处理的请求后缀:可以通过常量struts.action.extension进行修改,如:

      <struts>

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

      </struts>

     (4)

    <!-- 动态的给Action类的属性赋值 -->
        <package name="ztq" namespace="/test" extends="struts-default">
            <action name="a1" class="me.ztq.action.HelloWorldAction">
                <param name="message">哈哈</param>
                <result name="success">/3.jsp</result>
            </action>
        </package>
    <!-- 获取Action的值 -->
        <package name="ztq" namespace="/test" extends="struts-default">
            <action name="a2" class="me.ztq.action.HelloWorldAction" method="sayHello">
                <result type="redirect" name="success">/3.jsp?msg=${message}</result>
            </action>
        </package>

     (5)全局结果视图配置:

      <package name="mypackage" extends="struts-default">
            <!-- 配置全局错误结果:范围只是本包 -->
            <global-results>
                <result type="dispatcher" name="error">/customer/error.jsp</result>
            </global-results>
        </package>
        
        <package name="customer" namespace="/customer" extends="mypackage">
            <action name="addCustomer" class="me.ztq.action.CustomerAction" method="addCustomer">
                <result type="dispatcher" name="success">/customer/success.jsp</result>
            </action>
            <action name="updateCustomer" class="me.ztq.action.CustomerAction" method="updateCustomer">
                <result type="dispatcher" name="success">/customer/success.jsp</result>
            </action>
        </package>

     (6)struts2中的常量及设置:

    4. 动作类的生命周期:

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

    5. 为应用指定多个struts配置文件:

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

    <struts>

      <include file="struts-user.xml" />

      <include file="struts-order.xml" />

    </struts>

    6. 动态方法调用(不建议使用):

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

    假设访问action的URL路径为:/struts/test/helloworld.action

    如果要访问其中的other()方法,可以调用:/struts/test/helloworld!other.action

    使用通配符进行动态方法调用:

       <package name="orders" namespace="/orders" extends="mypackage">
            <action name="order_*" class="me.ztq.action.OrderAction" method="{1}">
                <result type="dispatcher" name="success">/orders/{1}.jsp</result>
            </action>
        </package>

    http://localhost:8080/struts/orders/order_add    就会访问add方法,转到add.jsp页面

    7. 获取表单请求参数:

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

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

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

     1 public class ProductAction{
     2     private Integer id;
     3     
     4     //struts2通过反射技术调用与请求参数同名的属性的setter方法来获取请求属性值
     5     public void setId(Integer id){
     6         this.id = id;
     7     }
     8     public Integer getId(){
     9         return id;
    10     }
    11 }

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

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

     1 public class ProductAction{
     2     private ProductAction product;
     3     private Integer id;
     4     public Integer getId() {
     5         return id;
     6     }
     7 
     8     public void setId(Integer id) {
     9         this.id = id;
    10     }
    11 
    12     public ProductAction getProduct() {
    13         return product;
    14     }
    15 
    16     public void setProduct(ProductAction product) {
    17         this.product = product;
    18     }
    19     
    20 }

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

    注:关于struts2.1.6接收中文请求参数乱码问题

    struts2.1.6版本中存在一个bug,即接收到的中文请求参数为乱码(以post方式提交),原因是struts2.1.6在获取并使用了请求参数后才调用HttpServletRequest的setCharacterEncoding()方法进行编码设置,导致应用使用的就是乱码请求参数。这个bug在struts2.1.8中已经被解决。如使用的是2.1.6,要解决这个问题,可以新建一个Filter,把这个Filter放置在struts2的Filter之前,然后在doFilter()方法里添加如下代码:

    1 public void doFilter(...){
    2     HttpServletRequest req = (HttpServletRequest)request;
    3     req.setCharacterEncoding("UTF-8");
    4     filterchain.doFilter(request, response);
    5 }

    8. 自定义类型转换器:

    (1)编写一个类,继承com.opensymphony.xwork2.conversion.impl.DefaultTypeConverter

    (2)覆盖掉其中的public Object convertValue(Map<String, Object> context, Object value,Class toType)

     1 public class DateConvertor extends DefaultTypeConverter {
     2         /*
     3          context:ognl表达式的上下文
     4          value:用户输入的值( 保存数据时)或者模型中的属性。用户输入的值是String数组
     5          toType:目标类型
     6          */
     7         @Override
     8         public Object convertValue(Map<String, Object> context, Object value,
     9                 Class toType) { 
    10             DateFormat df = new SimpleDateFormat("yyyy/MM/dd");
    11             if(toType==Date.class){
    12                 //2017/04/02----->java.util.Date 保存数据时
    13                 String strValue = ((String[])value)[0];
    14                 try {
    15                     return df.parse(strValue);
    16                 } catch (ParseException e) {
    17                     throw new RuntimeException(e);
    18                 }
    19             }else{
    20                 //java.util.Date----->2017/04/02 获取数据时
    21                 Date dValue = (Date)value;
    22                 return df.format(dValue);
    23             }
    24         }
    25     }

    (3)注册类型转换器

      ○ 局部类型转换器:只对当前的Action有效

        具体做法:在动作类相同的包中,建立一个名称是“动作类名-conversion.properties”的配置文件,文件中增加以下内容:要验证的字段=验证器的类全名,如:birthday=cn.itcast.convertor.DateConvertor

      ○ 全局类型转换器:对所有的Action都有效

        具体做法:在WEB-INF/classes目录下,建立一个名称为"xwork-conversion.properties"的配置文件,文件中增加以下内容:目标类型全名=验证器的类全名,如:java.util.Date=cn.itcast.convertor.DateConvertor

    注意:如果转换失败,Struts2框架会寻找name=input的结果页面

    9. 获取Servlet相关对象

    (1)通过ServletActionContext类直接获取:

    1 public String execute(){
    2     ServletContext sc = ServletActionContext.getServletContext();
    3     ServletRequest request = ServletActionContext.getRequest();
    4     return SUCCESS;
    5 }

    (2)实现指定接口,由struts框架运行时注入:

     1 public class testAction extends ActionSupport implements Serializable, ServletContextAware, ServletRequestAware, ServletResponseAware{
     2     private ServletContext context;
     3     private HttpServletRequest request;
     4     private HttpServletResponse response;
     5     public String execute(){
     6         System.out.println(context);
     7         System.out.println(request);
     8         System.out.println(response);
     9     }
    10     
    11     //如果动作类实现了ServletContextAware接口,就会自动调用该方法
    12     public void setServletContext(ServletContext context){
    13         this.context = context;
    14     }
    15     
    16     public void setServletRequest(HttpServletRequest request){
    17         this.request = request;
    18     }
    19     
    20     public void setServletResponse(HttpServletResponse response){
    21         this.response = response;
    22     }
    23 }

    10. 文件上传

    第一步、在WEB-INF/lib下加入commons-fileupload-1.2.1.jar、commons-io-1.3.2.jar两个文件

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

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

      <input type="file" name="uploadImage" />

    </form>

    第三步、在Action类中添加以下属性:

     1 public class HelloWorldAction{
     2     private File uploadImage; //得到上传的文件
     3     private String uploadImageContentType;  //得到文件的类型
     4     private String uploadImageFileName;  //得到文件的名称
     5     //这里省略了属性的getter/setter方法
     6     
     7     public String upload() throws Exception{
     8         String realPath = ServletActionContext.getServletContext().getRealPath("/images");
     9         File file = new File(realPath);
    10         if(!file.exists())
    11             file.mkdirs();
    12         FileUtils.copyFile(uploadImage, new File(file, uploadImageFileName));
    13         return "success";
    14     }
    15 }

    11. 自定义拦截器

    (1)编写一个类,实现com.opensymphony.xwork2.interceptor.Interceptor

    (2)主要实现public String intercept(ActionInvocation invocation) throws Exception{}方法,该方法的返回值就相当于动作的返回值。如果调用了String result = invocation.invoke(),就得到了动作类的返回的值。

     1 public String intercept(ActionInvocation invocation) throws Exception {
     2         //判断用户是否登录
     3         HttpSession session = ServletActionContext.getRequest().getSession();
     4         Object obj = session.getAttribute("user");
     5         if(obj==null){
     6             return "login";
     7         }else{
     8             return invocation.invoke();//调用动作方法
     9         }
    10     }

    (3)拦截器定义好后,一定要在配置文件中进行注册:

    <interceptors> <!--只是定义拦截器,并没有起作用--> 
      <interceptor name="permissionInterceptor" class="cn.itcast.interceptor.PermissionInterceptor"></interceptor>
    </interceptors>

    (4)配置文件中的动作,要通过<interceptor-ref name="permissionInterceptor"></interceptor-ref>使用该拦截器。

      注意:一旦动作中使用了自定义的拦截器,那么默认的就不起作用了。一般应该采用如下的做法:

      <interceptor-ref name="defaultStack"></interceptor-ref>
      <interceptor-ref name="permissionInterceptor"></interceptor-ref>

            <interceptors>
                <interceptor name="permission" class="me.ztq.action.PermissionInterceptor" />
                <interceptor-stack name="permissionStack">
                    <interceptor-ref name="defaultStack" />
                    <interceptor-ref name="permission" />
                </interceptor-stack>
            </interceptors>

      多个动作类都要使用的话,可以通过package来进行组合。

    因为struts2中如文件上传,数据验证,封装请求参数到action等功能都是由系统默认的defaultStack中的拦截器实现的,所以我们定义的拦截器需要引用系统默认的defaultStack,这样应用才可以使用struts2框架提供的众多功能。

    如果希望包下的所有action都使用自定义的拦截器,可以通过<default-interceptor-ref name="permissionStack" />把拦截器定义为默认拦截器。注意:每个包只能指定一个默认拦截器。另外,一旦我们为该包中的某个action显式指定了某个拦截器,则默认拦截器不会起作用。

    13. 用户输入数据验证

    在struts2中,我们可以实现对action的所有方法进行校验或者对action的指定方法进行校验。

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

    (1)采用手工编写代码实现,针对该动作类中的所有动作方法

      动作类的所有方法进行验证:

    • 动作类继承ActionSupport类
    • 覆盖掉public void validate()方法
    • 在validate方法中,编写不符合要求的代码判断,并调用父类的addFieldError(String fieldName, String errorMessage)方法,如果fieldError(存放错误信息的Map)有任何的元素,就是验证不通过,动作方法不会执行。struts2框架会返回到name=input的result
    • 在name=result指定的页面上使用struts2的标签显示错误信息<%@ taglib uri="/struts-tags" prefix="s"%>       <s:fielderror />

      动作类的指定方法进行验证:

    • 编写步骤与上面相同
    • 验证方法书写有要求:public void validateXxx()   Xxx代表的是要验证的动作方法名,其中要把动作方法名的首字母变为大写。

    (2)基于XML配置方式实现

      动作类的所有方法进行验证:

        使用基于XML配置方式实现输入校验时,Action也需要继承ActionSupport,并且提供校验文件,校验文件和Action类放在同一个包中,文件的取名格式为:ActionClassName-validation.xml,其中ActionClassName为Action的简单类名,-validation为固定写法。如果Action类为me.ztq.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">
         <!-- 内置验证器都是定义好的,在xwork-core.jar com.opensymphony.xwork2.validator.validators包中的default.xml文件中 -->
         <field-validator type="requiredstring"><!-- 不能为null或者""字符串,默认会去掉前后的空格 -->
            <message>用户名不能为空</message>
          </field-validator>
       </field>
    </validators>
        <field name="password">
            <field-validator type="requiredstring">
                <message>密码不能为空</message>
            </field-validator>
            <field-validator type="regex">
                <param name="expression"><![CDATA[d{3, 6}]]></param>
                <message>密码必须是3~6位数字</message>
            </field-validator>
        </field>

      动作类的指定方法进行验证:

        配置文件的名称书写有一定要求:动作类名-动作名(配置文件中的)-validation.xml

     (3)自定义基于XML的验证器:

    • 编写一个类,继承FieldValidatorSupport类
    • 在public void validate(Object object)编写验证逻辑,不符合要求的就向fieldErrors中放消息
    • 一定要注册验证器后才能使用,在WEB-INF/classes目录下建立一个名称为validators.xml的配置文件,内容如下:
    <validators>
      <validator name="strongpassword" class="cn.itcast.validators.StrongPasswordValidator"/>
    </validators>
    • 以后就可以像使用Struts2提供的16个验证器方式去使用了。

     StrongPasswordValidator.class

     1 package me.ztq.validators;
     2 
     3 import java.io.Serializable;
     4 
     5 import com.opensymphony.xwork2.validator.ValidationException;
     6 import com.opensymphony.xwork2.validator.validators.FieldValidatorSupport;
     7 
     8 public class StrongPasswordValidator extends FieldValidatorSupport implements
     9         Serializable {
    10     
    11     private int minLength = -1;
    12     
    13     public int getMinLength() {
    14         return minLength;
    15     }
    16     
    17     public void setMinLength(int minLength) {
    18         this.minLength = minLength;
    19     }
    20     
    21     public void validate(Object object) throws ValidationException {
    22         String fieldName = getFieldName(); //取得字段名
    23         String fieldValue = (String)getFieldValue(fieldName, object); //取得用户输入的当前字段的值
    24         if(!isPasswordStrong(fieldValue)){
    25             addFieldError(fieldName, object);
    26         }
    27     }
    28     
    29     private static final String GROUP1 = "abcdefghijklmnopqrstuvwxyz";
    30     private static final String GROUP2 = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
    31     private static final String GROUP3 = "0123456789";
    32     private boolean isPasswordStrong(String password) {
    33         boolean ok1 = false;
    34         boolean ok2 = false;
    35         boolean ok3 = false;
    36         int length = password.length();
    37         for(int i = 0; i < length; i++){
    38             if(ok1 && ok2 && ok3)
    39                 break;
    40             String character = password.substring(i, i + 1);
    41             if(GROUP1.contains(character)){
    42                 ok1 = true;
    43                 continue;
    44             }
    45             if(GROUP2.contains(character)){
    46                 ok2 = true;
    47                 continue;
    48             }
    49             if(GROUP3.contains(character)){
    50                 ok3 = true;
    51                 continue;
    52             }
    53         }
    54         return ok1 && ok2 && ok3;
    55     }
    56 
    57 }

    14. 国际化

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

    baseName_language_country.properties

    baseName_language.properties

    baseName.properties

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

    中国大陆:baseName_zh_CN.properties

    美国:baseName_en_US.properties

    (1)全局资源文件:放到WEB-INF/classes目录下

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

    <constant name="struts.custom.i18n.resources" value="me" /> me为资源文件的基本名。

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

    • 在JSP页面中使用<s:text name="" />标签输出国际化信息:<s:text name="user" />,name为资源文件中的key
    • 在Action类中,可以继承ActionSupport,使用getText()方法得到国际化信息,该方法的第一个参数用于指定资源文件中的key
    • 在表单标签中,通过key属性指定资源文件中的key,如:<s:textfield name="realname" key="user" />

    (2)包范围资源文件:服务于Java类的包下的动作类

      取名:package_语言_国家.properties

      在Java的包下放置package_语言_国家.properties资源文件,package为固定写法,处于该包及子包下的action都可以访问该资源。当查找指定key的消息时,系统会从package资源文件查找,当找不到对应的key时,才会从常量struts.i18n.resources指定的资源文件中寻找。

    (3)Action范围资源文件:

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

      当查找指定key的消息时,系统会先从ActionClassName_language_country.properties资源文件查找,如果没有找到对应的key,然后沿着当前包往上查找基本名为package的资源文件,一直找到最顶层包。如果还没有对应的key,最后从常量struts.i18n.resources指定的资源文件中寻找。

    在JSP中直接访问某个资源文件:

    struts2提供了<s:i18n>标签,使用<s:i18n>标签我们可以再类路径下直接从某个资源文件中获取国际化数据,而无需任何配置: 

    <s:i18n name="ztq">

      <s:text name="welcome" />

    </s:i18n>

    ztq为类路径下资源文件的基本名。如果要访问的资源文件在类路径的某个包下,可以这样访问:

    <s:i18n name="me/ztq/action/package">

      <s:text name="welcome">

        <s:param>呵呵</s:param>

      </s:text> 

    </s:i18n>

    上面访问me.ztq.action包下基本名为package的资源文件。

    15. OGNL表达式

    (1)OGNL是Object Graphic Navigation Language(对象图导航语言)的缩写,它是一个开源项目。Struts2框架使用OGNL作为默认的表达式语言。

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

    • 支持对象方法调用,如:XXX.sayHello();
    • 支持类静态方法调用和值访问,表达式的格式为@类全名(包括包路径)@方法名(或者值名)。例如:@java.lang.String@format('foo %s', 'bar')或@me.ztq.Constant@APP_NAME
    • 操作集合对象
    • OGNL有一个上下文(Context)概念,就是一个Map结构,它实现了java.utils.Map接口,struts2中上下文(Context)的实现为ActionContext。
    • 当struts接受一个请求时,会迅速创建ActionContext,ValueStack,action。然后把action存放进ValueStack,所以action的实例变量可以被OGNL访问。
    • 访问上下文(Context)中的对象需要使用#符号标注命名空间,如:#application、#session
    • 另外OGNL会设定一个根对象(root对象),在struts2中根对象就是ValueStack(值栈)。访问根对象(即ValueStack)中对象的属性,则可以省略#命名空间,直接访问该对象的属性即可。
    • 在Struts2中,根对象ValueStack的实现类为OgnlValueStack,该对象不是我们想象的只存放单个值,而是存放一组对象。在OgnlValueStack类里有一个List类型的root变量,就是使用它存放一组对象。
    • Context----OnglValueStack root变量[action, OgnlUtil, ...]
    • 在root变量中处于第一位的对象叫栈顶对象。通常我们在OGNL表达式里直接写上属性的名称即可访问root变量里对象的属性,搜索顺序是从栈顶对象开始寻找,如果栈顶对象不存在该属性,就会从第二个对象寻找,如果没有找到就从第三个对象寻找,依次往下访问,直到找到为止。

    综上:

    ActionContext的结构:

        ValueStack:

          List:动作类放在此处。取存放在ValueStack中的root的对象的属性,直接写即可。访问以下内容中的对象要使用#+范围

        application:用于访问ServletContext,例如:#application.userName或者application['userName'],相当于调用ServletContext的getAttribute("username")。

        session:用于访问HttpSession

        request:用于访问HttpServletRequest

        parameters:用于访问HTTP的请求参数

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

    注:struts2中,OGNL表达式需要配合struts标签才可以使用,如:<s:property value="name" />;除此之外,在页面中可以使用<s:debug />查看上下文中的对象

    (2)采用OGNL表达式创建List/Map集合对象:

    如果需要一个集合元素的时候(例如List对象或者Map对象),可以使用OGNL中同集合相关的表达式。使用如下代码直接生成一个List对象:

    <s:set var="list" value="{'a', 'b', 'c'}" />

    <s:iterator value="#list">

      <s:property /><br />

    </s:iterator>

    Set标签用于将某个值放入指定范围。

    scope:指定变量被放置的范围,该属性可以接受application、session、request、page或action。如果没有设置该属性,则默认放置在OGNL Context中。

    value:赋给变量的值。如果没有设置该属性,则将ValueStack栈顶的值赋给变量。

    生成一个Map对象:

    <s:set name="foobar" value="#{'foo1':'bar1', 'foo2':'bar2'}" />

    <s:iterator value="#foobar" var="me">

      <s:property value="#me.key" />=<s:property value="#me.value" /><br />

    </s:iterator>

     也可以:

    <c:forEach items="${foobar}" var="me">

      ${me.key} = ${me.value} <br />

    </c:forEach>

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

    对于集合类型,OGNL表达式可以使用in和not 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>

    (4)OGNL表达式的投影功能

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

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

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

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

    如:

    <s:iterator value="books.{?#his.price>35}">

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

    </s:iterator>

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

    1 public class BookAction extends ActionSupport{
    2     private List<Book> books;
    3     public String execute(){
    4         books = new ArrayList<Book>();
    5         books.add(new Book("a", "spring", 67));
    6         books.add(new Book("b", "ejb", 15));
    7     }
    8 }

    16. 常用标签:

    (1)property

      property标签用于输出指定值:

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

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

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

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

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

      id可选属性,指定该元素的标识。(过时)

    (2)Iterator标签用于对集合进行迭代,这里的集合包含List、Set和数组

        <s:set name="list" value="{'a', 'b', 'c'}" />
        <s:iterator value="#list" status="st">
        <font color=<s:if test="#st.odd">red</s:if><s:else>blue</s:else>>
        <s:property /></font><br />
        </s:iterator>

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

    id:可选属性,指定该元素的标识。(过时)

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

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

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

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

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

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

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

    如:

    <body>
        <s:set var="records" value="{'a', 'b', 'c', 'd'}" />
        <table border="1">
            <tr>
                <th>序号</th>
                <th>书名</th>
            </tr>
            <s:iterator value="records" status="st">
                <tr>
                    <td>
                        <s:property value="#st.count" />
                    </td>
                    <td>
                        <s:property />
                    </td>
                </tr>
            </s:iterator>
        </table>
     </body>

    (3)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.haha.net" />

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

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

    输出结果:

    #myurl

    http://www.haha.net

    (4)checkboxlist复选框

    如果集合为list

    <s:checkboxlist name="list" list="{'Java', '.Net', 'RoR', 'PHP'}" value="{'Java', '.Net'}" />

    生成如下Html代码:

    <input type="checkbox" name="list" value="Java" checked="checked" /><label>Java</label>

    <input type="checkbox" name="list" value=".Net" checked="checked" /><label>.Net</label>

    <input type="checkbox" name="list" value="RoR" /><label>RoR</label>

    <input type="checkbox" name="list" value="PHP" /><label>PHP</label>

    如果集合为Map

    <s:checkboxlist name="map" list="#{1:'瑜伽用品', 2:'户外用品', 3:'球类', 4:'自行车'}" listKey="key" listValue="value" value="{1, 2, 3}" />

    生成如下html代码:

    <input type="checkbox" name="map" value="1" checked="checked" /><label>瑜伽用品</label>

    <input type="checkbox" name="map" value="2" checked="checked" /><label>户外用品</label>

    <input type="checkbox" name="map" value="3" checked="checked" /><label>球类</label>

    <input type="checkbox" name="map" value="4" /><label>自行车</label>

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

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

      <body>
        <s:form action="addCustomer" namespace="/customer">
            <s:token></s:token>
            <s:textfield name="username" label="用户名" />
            <s:submit value="保存" />
        </s:form>
      </body>
            <action name="addCustomer" class="me.ztq.action.CustomerAction" method="add">
                <interceptor-ref name="defaultStack"></interceptor-ref>
                <interceptor-ref name="token"></interceptor-ref>
                <result name="invalid.token">/success.jsp</result>
                <result name="success">/success.jsp</result>
            </action>

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

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

  • 相关阅读:
    闭包
    递归 斐波那契
    作用域,函数的四种调用模式
    数组方法
    Math内置对象
    BeanUtils.copyProperties() 用法
    ToStringBuilder(this,ToStringStyle.MULTI_LINE_STYLE) .append("Id",getId())防止内存泄漏
    Spring配置文件applicationContext.xml Hibernate SQL方言 (hibernate.dialect)
    Redis面试总结
    Nginx面试题
  • 原文地址:https://www.cnblogs.com/zhangtianq/p/6591631.html
Copyright © 2011-2022 走看看