java深入探究12-框架之Structs

注意每次修改配置文件后必须项目重启

Structs2=structs1+xwork

Struct2框架预先实现了一些功能:

  1.请求数据的封装;2.文件上传的功能3.对国际化功能的简化4.文件效验功能

1.开发Structs框架的步骤:

  1)引入8大jar包

    commons-fileupload-1.2.2.jar   【文件上传相关包】

    commons-io-2.0.1.jar

    struts2-core-2.3.4.1.jar           struts2核心功能包】

    xwork-core-2.3.4.1.jar           Xwork核心包】

    ognl-3.0.5.jar  Ognl表达式功能支持表】

    commons-lang3-3.1.jar          strutsjava.lang包的扩展】

    freemarker-2.3.19.jar            struts的标签模板库jar文件】

    javassist-3.11.0.GA.jar           struts对字节码的处理相关jar

  2)配置web.xml

    主要配置filter Struct过滤器,StructsPrepareAndExecuteFilter核心过滤器

   //引入struct核心过滤器
    <filter>
        <filter-name>struct2</filter-name>
        <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>struct2</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>

  3)开发Action 

    方法满足:无参数,放回值为String,public修饰

编写普通类1.继承ActionSupport有数据效验时必须继承;2.或者实现action接口
     3.什么都不写,直接return ”success”再从structs.xml中找到对应的返回页面
例如:

public String register() {
System.out.println("register()" + userName);
return "register";
}

再配置文件中配置

<action name="register" class="cn.itcast.a_config.UserAction" method="register">
<result name="success">/index.jsp</result>
</action>

可以使用通配符优化配置

<!-- 使用通配符优化上面的步骤 -->
<!-- http://localhost:8080/struts02/user_login -->
<action name="user_*" class="cn.itcast.a_config.UserAction" method="{1}">
<result name="{1}">/{1}.jsp</result>
</action>


public class HelloAction extends ActionSupport {
    // 处理请求
    public String execute() throws Exception {}
}

  4)配置struct.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
          "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
          "http://struts.apache.org/dtds/struts-2.0.dtd">
<struts>
    <!-- 
        package   定义一个包。 包作用,管理action。
                   (通常,一个业务模板用一个包)
           name  包的名字; 包名不能重复;
           extends 当前包继承自哪个包
                      在struts中,包一定要继承struts-default
                     struts-default在struts-default.xml中定的包
           abstract  
                     表示当前包为抽象包; 抽象包中不能有action的定义,否则运行时期报错
                    abstract=true  只有当当前的包被其他包继承时候才用!
                    如:
                        <package name="basePackage" extends="struts-default" abstract="true"></package>
                       <package name="user" extends="basePackage">
              namespace   名称空间,默认为"/"
                              作为路径的一部分
                             访问路径=  http://localhost:8080/项目/名称空间/ActionName
              
              
          
        action   配置请求路径与Action类的映射关系
                 name  请求路径名称
                 class 请求处理的aciton类的全名
                 method 请求处理方法        
        result
                 name  action处理方法返回值 
                 type  跳转的结果类型
                   标签体中指定跳转的页面
                    
     -->
    <package name="user" extends="struts-default" namespace="/">
        <action name="login" class="cn.itcast.b_execute.UserAction" method="login">
            <result name="login">/index.jsp</result>
        </action>
    </package> 
    
</struts>

2.访问流程:

   tomcat服务器启动-》读取web,xml-》读取struct2核心过滤器-》初始化过滤器-》init方法(这里分别读取了struct-default.xml核心功能初始化有拦截器等;struct-plugin.xml:struct相关插件;struct.xml用户编写的xml)-》读取到struct.xml后找到action类-》读取structs.properties用户自定义配置文件会覆盖Structs.xml中的常量设置-》加载到内存中等待访问再实例化action类

3.一些配置文件详解

 struct-default.xml

目录:struts2-core-2.3.4.1.jar/ struts-default.xml
 内容:
    1. bean节点指定struts在运行的时候创建的对象类型
    2.指定struts-default包  【用户写的package(struts.xml)一样要继承此包 】
        package  struts-default  包中定义了:
            a.  跳转的结果类型
                dispatcher    转发,不指定默认为转发
                redirect       重定向
                redirectAction  重定向到action资源
                stream        (文件下载的时候用)
            b. 定义了所有的拦截器
                  定义了32个拦截器!
                  为了拦截器引用方便,可以通过定义栈的方式引用拦截器,
                此时如果引用了栈,栈中的拦截器都会被引用!
                
                defaultStack
                    默认的栈,其中定义默认要执行的18个拦截器!


            c. 默认执行的拦截器栈、默认执行的action
                <default-interceptor-ref name="defaultStack"/>
               <default-class-ref class="com.opensymphony.xwork2.ActionSupport" />


<interceptor 
name="prepare" class="com.opensymphony.xwork2.interceptor.PrepareInterceptor"/>
<interceptor 
name="params" class="com.opensymphony.xwork2.interceptor.ParametersInterceptor"/>
View Code

自己的struct。xml配置

  1)两种方便访问action的方法“通配符”“动态配置”

   通配符:可以使用* 和{1}来优化配置

   动态配置:这个访问方式action名字!action类中的需要访问的方法名例如:hello!add.action

    /struct2/hello!add.action:用这个只有在常量设置中设置

    <constant name="struts.enable.DynamicMethodInvocation" value="false"/>
<action name="user_*_*" class="" method="{1}{2}">
    <result name="{1}">/{1}.jsp</result>
</action>

  2)路径匹配原则

       /Struts2_01/hello_a/a/b/helloWorld.action

       /Struts2_01/hello_a/a/b找package->没找到/Struts2_01/hello_a/a-》没找到/Struts2_01/hello_a没找到/Struts2_01/-报404错

  3)常量

    所有的初始化全局变量配置都在Structs-core-2.3.4-1.jar/org.apache.structs/default.properities

#
# $Id: default.properties 1132110 2011-06-05 08:45:32Z lukaszlenart $
#
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements.  See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership.  The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License.  You may obtain a copy of the License at
#
#  http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing,
# software distributed under the License is distributed on an
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
# KIND, either express or implied.  See the License for the
# specific language governing permissions and limitations
# under the License.
#
### START SNIPPET: complete_file

### Struts default properties
###(can be overridden by a struts.properties file in the root of the classpath)
###

### Specifies the Configuration used to configure Struts
### one could extend org.apache.struts2.config.Configuration
### to build one's customize way of getting the configurations parameters into Struts
# struts.configuration=org.apache.struts2.config.DefaultConfiguration

### This can be used to set your default locale and encoding scheme
# struts.locale=en_US
struts.i18n.encoding=UTF-8

### if specified, the default object factory can be overridden here
### Note: short-hand notation is supported in some cases, such as "spring"
###       Alternatively, you can provide a com.opensymphony.xwork2.ObjectFactory subclass name here
# struts.objectFactory = spring

### specifies the autoWiring logic when using the SpringObjectFactory.
### valid values are: name, type, auto, and constructor (name is the default)
struts.objectFactory.spring.autoWire = name

### indicates to the struts-spring integration if Class instances should be cached
### this should, until a future Spring release makes it possible, be left as true
### unless you know exactly what you are doing!
### valid values are: true, false (true is the default)
struts.objectFactory.spring.useClassCache = true

### ensures the autowire strategy is always respected.
### valid values are: true, false (false is the default)
struts.objectFactory.spring.autoWire.alwaysRespect = false

### if specified, the default object type determiner can be overridden here
### Note: short-hand notation is supported in some cases, such as "tiger" or "notiger"
###       Alternatively, you can provide a com.opensymphony.xwork2.util.ObjectTypeDeterminer implementation name here
### Note: By default, com.opensymphony.xwork2.util.DefaultObjectTypeDeterminer is used which handles type detection
###       using generics. com.opensymphony.xwork2.util.GenericsObjectTypeDeterminer was deprecated since XWork 2, it's
###       functions are integrated in DefaultObjectTypeDeterminer now.
###       To disable tiger support use the "notiger" property value here.
#struts.objectTypeDeterminer = tiger
#struts.objectTypeDeterminer = notiger

### Parser to handle HTTP POST requests, encoded using the MIME-type multipart/form-data
# struts.multipart.parser=cos
# struts.multipart.parser=pell
struts.multipart.parser=jakarta
# uses javax.servlet.context.tempdir by default
struts.multipart.saveDir=
struts.multipart.maxSize=2097152

### Load custom property files (does not override struts.properties!)
# struts.custom.properties=application,org/apache/struts2/extension/custom

### How request URLs are mapped to and from actions
#struts.mapper.class=org.apache.struts2.dispatcher.mapper.DefaultActionMapper

### Used by the DefaultActionMapper
### You may provide a comma separated list, e.g. struts.action.extension=action,jnlp,do
### The blank extension allows you to match directory listings as well as pure action names
### without interfering with static resources, which can be specified as an empty string
### prior to a comma e.g. struts.action.extension=, or struts.action.extension=x,y,z,,
struts.action.extension=action,,

### Used by FilterDispatcher
### If true then Struts serves static content from inside its jar.
### If false then the static content must be available at <context_path>/struts
struts.serve.static=true

### Used by FilterDispatcher
### This is good for development where one wants changes to the static content be
### fetch on each request.
### NOTE: This will only have effect if struts.serve.static=true
### If true -> Struts will write out header for static contents such that they will
###             be cached by web browsers (using Date, Cache-Content, Pragma, Expires)
###             headers).
### If false -> Struts will write out header for static contents such that they are
###            NOT to be cached by web browser (using Cache-Content, Pragma, Expires
###            headers)
struts.serve.static.browserCache=true

### Set this to false if you wish to disable implicit dynamic method invocation
### via the URL request. This includes URLs like foo!bar.action, as well as params
### like method:bar (but not action:foo).
### An alternative to implicit dynamic method invocation is to use wildcard
### mappings, such as <action name="*/*" method="{2}" class="actions.{1}">
struts.enable.DynamicMethodInvocation = true

### Set this to true if you wish to allow slashes in your action names.  If false,
### Actions names cannot have slashes, and will be accessible via any directory
### prefix.  This is the traditional behavior expected of WebWork applications.
### Setting to true is useful when you want to use wildcards and store values
### in the URL, to be extracted by wildcard patterns, such as
### <action name="*/*" method="{2}" class="actions.{1}"> to match "/foo/edit" or
### "/foo/save".
struts.enable.SlashesInActionNames = false

### use alternative syntax that requires %{} in most places
### to evaluate expressions for String attributes for tags
struts.tag.altSyntax=true

### when set to true, Struts will act much more friendly for developers. This
### includes:
### - struts.i18n.reload = true
### - struts.configuration.xml.reload = true
### - raising various debug or ignorable problems to errors
###   For example: normally a request to foo.action?someUnknownField=true should
###                be ignored (given that any value can come from the web and it
###                should not be trusted). However, during development, it may be
###                useful to know when these errors are happening and be told of
###                them right away.
struts.devMode = false

### when set to true, resource bundles will be reloaded on _every_ request.
### this is good during development, but should never be used in production
struts.i18n.reload=false

### Standard UI theme
### Change this to reflect which path should be used for JSP control tag templates by default
struts.ui.theme=xhtml
struts.ui.templateDir=template
#sets the default template type. Either ftl, vm, or jsp
struts.ui.templateSuffix=ftl

### Configuration reloading
### This will cause the configuration to reload struts.xml when it is changed
struts.configuration.xml.reload=false

### Location of velocity.properties file.  defaults to velocity.properties
struts.velocity.configfile = velocity.properties

### Comma separated list of VelocityContext classnames to chain to the StrutsVelocityContext
struts.velocity.contexts =

### Location of the velocity toolbox
struts.velocity.toolboxlocation=

### used to build URLs, such as the UrlTag
struts.url.http.port = 80
struts.url.https.port = 443
### possible values are: none, get or all
struts.url.includeParams = none

### Load custom default resource bundles
# struts.custom.i18n.resources=testmessages,testmessages2

### workaround for some app servers that don't handle HttpServletRequest.getParameterMap()
### often used for WebLogic, Orion, and OC4J
struts.dispatcher.parametersWorkaround = false

### configure the Freemarker Manager class to be used
### Allows user to plug-in customised Freemarker Manager if necessary
### MUST extends off org.apache.struts2.views.freemarker.FreemarkerManager
#struts.freemarker.manager.classname=org.apache.struts2.views.freemarker.FreemarkerManager

### Enables caching of FreeMarker templates
### Has the same effect as copying the templates under WEB_APP/templates
struts.freemarker.templatesCache=false

### Enables caching of models on the BeanWrapper
struts.freemarker.beanwrapperCache=false

### See the StrutsBeanWrapper javadocs for more information
struts.freemarker.wrapper.altMap=true

### maxStrongSize for MruCacheStorage for freemarker
struts.freemarker.mru.max.strong.size=100

### configure the XSLTResult class to use stylesheet caching.
### Set to true for developers and false for production.
struts.xslt.nocache=false

### Whether to always select the namespace to be everything before the last slash or not
struts.mapper.alwaysSelectFullNamespace=false

### Whether to allow static method access in OGNL expressions or not
struts.ognl.allowStaticMethodAccess=false

### Whether to throw a RuntimeException when a property is not found
### in an expression, or when the expression evaluation fails
struts.el.throwExceptionOnFailure=false

### Logs as Warnings properties that are not found (very verbose)
struts.ognl.logMissingProperties=false

### Caches parsed OGNL expressions, but can lead to memory leaks
### if the application generates a lot of different expressions
struts.ognl.enableExpressionCache=true
### END SNIPPET: complete_file
View Code

    1.Struct中默认访问后缀

      Struct1:.do;Struct2:.action

    2.在Struct.xml通过常量修改

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

      指定访问后缀为action/do/没有访问后缀都可以

      value="action,do,"不带后缀

      value="action,do"访问后缀action或do

      value="action" 后缀只能是action

    3.常量在struct.xml中配置

 <!-- 1.常量 -->
    <!-- 指定默认编码集,作用于HttpServletRequest的setCharacterEncoding方法 和freemarker 、velocity的输出 -->
    <constant name="struts.i18n.encoding" value="UTF-8"/>
    <!-- 自定义后缀修改常量 -->
    <constant name="struts.action.extension" value="do"/>
    <!-- 设置浏览器是否缓存静态内容,默认值为true(生产环境下使用),开发阶段最好关闭 --> 
    <constant name="struts.serve.static.browserCache" value="false"/>
    <!-- 当struts的配置文件修改后,系统是否自动重新加载该文件,默认值为false(生产环境下使用),开发阶段最好打开 --> 
    <constant name="struts.configuration.xml.reload" value="true"/>
    <!-- 开发模式下使用,这样可以打印出更详细的错误信息 --> 
    <constant name="struts.devMode" value="true" />
    <!-- 默认的视图主题  -->
    <constant name="struts.ui.theme" value="simple" />
    <!-- 与spring集成时,指定由spring负责action对象的创建 --> 
    <constant name="struts.objectFactory" value="spring" />
    <!-- 该属性设置Struts 2是否支持动态方法调用,该属性的默认值是true。如果需要关闭动态方法调用,则可设置该属性
    为 false -->
    <constant name="struts.enable.DynamicMethodInvocation" value="false"/>
    <!-- 上传文件的大小限制 -->
    <constant name="struts.multipart.maxSize" value="10701096"/>

  4)resultType

    1.在action中想要获得request对象:

      ServletActionContext.getRequest();ServletActionContext是action重要对象

    2.其中type的设置有

 

     3.result全局结果

      当多个action中使用了相同的result,为了避免result的重复,我们可以设置全局结果;但是局部权限大于全局

      

 4.数据封装

  1)struts对数据封装,当访问action时,参与核心过滤器,访问default-struts.xml->strut.xml,在default-struts.xml中有32个拦截器,其中Parameters拦截器对数据进行封装

  2)String-》基本数据类型转换是自动的

    String-》Date日期类型的转换是有条件的

  3)转换原理过程

    1.表单中的name值自动映射到Action中的一个属性;2.还可以映射到一个集合

    3.Struts对HttpServetRequest,HttpSession,ServletContext进行了封装,构造了三个Map对象,可通过ServletActionContext来访问三个对象

  4)在action中两种获得数据的方法

    1.ServletApi 2.通过ServletActionContext获得ActionContext对象得到三个封装好的Map对象3.实现三个接口RequestAware, SessionAware, ApplicationAwar也可以

方式一:通过Servlet Api
HttpServletRequest request = ServletActionContext.getRequest(); HttpSession session = request.getSession(); ServletContext application = ServletActionContext.getServletContext(); // 操作 request.setAttribute("request_data", "request_data1"); session.setAttribute("session_data", "session_data1"); application.setAttribute("application_data", "application_data1");
方式二:推荐这个方法 
//
Struts中对数据操作,方式2: 通过ActionContext类 ActionContext ac = ActionContext.getContext(); // 得到Struts对HttpServletRequest对象进行了封装,封装为一个map // 拿到表示request对象的map Map<String,Object> request = ac.getContextMap(); // 拿到表示session对象的map Map<String, Object> session = ac.getSession(); // 拿到表示servletContext对象的map Map<String, Object> application = ac.getApplication(); // 数据 request.put("request_data", "request_data1_actionContext"); session.put("session_data", "session_data1_actionContext"); application.put("application_data", "application_data1_actionContext");
/**
 * 数据处理, 方式3: 实现接口的方法
 * @author Jie.Yuan
 *
 */
public class DataAction extends ActionSupport implements RequestAware, SessionAware, ApplicationAware{
    
    private Map<String, Object> request;
    private Map<String, Object> session;
    private Map<String, Object> application;
    
    // struts运行时候,会把代表request的map对象注入

    public void setRequest(Map<String, Object> request) {
        this.request = request;
    }
    
    // 注入session

    public void setSession(Map<String, Object> session) {
        this.session = session;
    }
    
    // 注入application

    public void setApplication(Map<String, Object> application) {
        this.application = application;
    }



    public String execute() throws Exception {
        
         // 数据
        /*
         * 
        // Struts中对数据操作,方式1: 直接拿到ServletApi, 执行操作
        HttpServletRequest request = ServletActionContext.getRequest();
        HttpSession session = request.getSession();
        ServletContext application = ServletActionContext.getServletContext();
        // 操作
        request.setAttribute("request_data", "request_data1");
        session.setAttribute("session_data", "session_data1");
        application.setAttribute("application_data", "application_data1");
        */
        
        // 【推荐:解耦的方式实现对数据的操作】
        // Struts中对数据操作,方式2: 通过ActionContext类 
        ActionContext ac = ActionContext.getContext();
        // 得到Struts对HttpServletRequest对象进行了封装,封装为一个map
        // 拿到表示request对象的map
        Map<String,Object> request =  ac.getContextMap(); 
        // 拿到表示session对象的map
        Map<String, Object> session = ac.getSession();
                 // 拿到表示servletContext对象的map
        Map<String, Object> application = ac.getApplication();
                 
        // 数据
        request.put("request_data", "request_data1_actionContext");
        session.put("session_data", "session_data1_actionContext");
        application.put("application_data", "application_data1_actionContext");
//        
        return SUCCESS;
    }

}
View Code

5.类型转换

  前面说了数据转换String->基本类型类型自动的,日期需要条件

  1)是Parameters拦截器做的类似于: Beanutils工具

  2)自定义类型转换器

    1.继承StrutsTypeConverter

    2.全局转换,局部转换配置

    例子:

/**
 * 自定义类型转换器类
 * 
 * @author Jie.Yuan
 * 
 */
public class MyConverter extends StrutsTypeConverter {

    // 新需求: 要求项目中要支持的格式,如: yyyy-MM-dd/yyyyMMdd/yyyy年MM月dd日..

    // 先定义项目中支持的转换的格式
    DateFormat[] df = { new SimpleDateFormat("yyyy-MM-dd"),
            new SimpleDateFormat("yyyyMMdd"),
            new SimpleDateFormat("yyyy年MM月dd日") };

    /**
     * 把String转换为指定的类型 【String To Date】
     * 
     * @param context
     *            当前上下文环境
     * @param values
     *            jsp表单提交的字符串的值
     * @param toClass
     *            要转换为的目标类型
     */
    @Override
    public Object convertFromString(Map context, String[] values, Class toClass) {

        // 判断: 内容不能为空
        if (values == null || values.length == 0) {
            return null;
        }
        // 判断类型必须为Date
        if (Date.class != toClass) {
            return null;
        }
        
        // 迭代:转换失败继续下一个格式的转换; 转换成功就直接返回
        for (int i=0; i<df.length; i++) {
            try {
                return df[i].parse(values[0]);
            } catch (ParseException e) {
                continue;
            }
        }
        return null;
    }

    @Override
    public String convertToString(Map context, Object o) {
        return null;
    }

}
View Code

    局部配置:在自定义转化器添加(自定义转换器名字-conversion.properties):MyConverter-conversion.properties

         在其中写:需要转换的字段名=自定义转换器类的权限定名birth=type.MyConverter

    全局配置:在项目src目录下建立固定文件xwork-conversion.properties

         在其中写:需要转换的类类型=转换器类的权限定名 :java.util.Date=type.MyConverter(java.util.Date是birth的类型)

  3)struts-default.xml <interceptor name="conversionError" class="org.apache.struts2.interceptor.StrutsConversionErrorInterceptor"/>

    该拦截器负责对错误信息处理

6.文件支持

  1)文件上传

     1.核心类:FileItemFactory;ServletFileUpload;FileItem

     2.struts处理上传文件:

        获得上传文件的file,对应的fileName,fileContextType,之后再execute中执行对上传文件的处理;

        上传文件默认都是被缓存到.me_tcatworkCatalinalocalhoststruts02upload_5bd0b60c_15c3369eeb3__8000_00000005.tmp中

        我们要做的是把文件得到存到我们指定的位置

     3.配置action:在action子栏中添加。。。等信息

      在default-struts中有拦截器fileUpload,而在ServletFileUpload类中文件属性我们都可以在自己写的拦截器中添加

<!-- 限制运行上传的文件的类型 -->
            <interceptor-ref name="defaultStack">
                
                <!-- 限制运行的文件的扩展名 -->
                <param name="fileUpload.allowedExtensions">txt,jpg,jar</param>
                <param name="fileUpload.maximumSize">1024</param>
                <!-- 限制运行的类型   【与上面同时使用,取交集】
                <param name="fileUpload.allowedTypes">text/plain</param>
                -->
                
            </interceptor-ref>  

    当文件上传错误时,会自动放回字符串input,我们可以在action中配置跳转到指定页面,这个页面想要显示错误信息可以通过引用struts标签,显示

    4.上传例子

1.jsp
<form action="${pageContext.request.contextPath }/fileUploadAction" method="post" enctype="multipart/form-data">
          用户名:<input type="text" name="userName"><br/>
          文件:<input type="file" name="file1"><br/>
          
          <input type="submit" value="上传">
      </form>
2.action
public class FileUpload extends ActionSupport {

    /**
     * 
     */
    private static final long serialVersionUID = 1L;
    // 对应表单:<input type="file" name="file1">
    private File file1; 
    // 文件名
    private String file1FileName;
    // 文件的类型(MIME)
    private String file1ContentType;
    public void setFile1(File file1) {
        this.file1 = file1;
    }
    public void setFile1FileName(String file1FileName) {
        this.file1FileName = file1FileName;
    }
    public void setFile1ContentType(String file1ContentType) {
        this.file1ContentType = file1ContentType;
    }
    //上面的代码是通过拦截器对数据自动封装到这三个属性中的
    
    @Override
    public String execute() throws Exception {
        /******拿到上传的文件,进行处理******/
        // 把文件上传到upload目录

        System.out.println(file1FileName);
        System.out.println(file1ContentType);
        // 获取上传的目录路径
        String path = ServletActionContext.getServletContext().getRealPath("/upload");
        System.out.println(path);
        // 创建目标文件对象
        File destFile = new File(path,file1FileName);
        // 把上传的文件,拷贝到目标文件中
        FileUtils.copyFile(file1, destFile);
        
        return SUCCESS;
    }
}
View Code   

  2)文件下载

    访问连接down_down.action就是到down_list的action中找到类,对应的list方法,下面的例子是list方法返回list字符串在action中就找到对于的result放回到list.jsp中,在这里通过点击下载-》又发送连接down_down?fileName=文件名.txt-》找到down_down的action找到对应的类里面的down方法开始下载-》随后在对应的action中找到result,这个result的为属于下载业务功能,需要特殊设置:

    type="stream";之后在子项中添加四个param子标签

<!-- 下载操作 -->
            <result name="download" type="stream">
               <!-- 运行下载的文件的类型:指定为所有的二进制文件类型 -->
               <param name="contentType">application/octet-stream</param>
               
               <!-- 对应的是Action中属性: 返回流的属性【其实就是getAttrInputStream()】 -->
               <param name="inputName">attrInputStream</param>
               
               <!-- 下载头,包括:浏览器显示的文件名 -->
               <param name="contentDisposition">attachment;filename=${downFileName}</param>
             
                 <!-- 缓冲区大小设置 -->
               <param name="bufferSize">1024</param>
            </result>

之后我们需要在action类中再写 1.放回流属性的getAttrInputStream()方法放回要下载文件的流;

               2.返回下载头包含浏览器显示的文件名:getDownFileName()

上传下载的整体例子

上传
1.file.xml中的配置
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
          "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
          "http://struts.apache.org/dtds/struts-2.0.dtd">
<struts>
    <package name="file" extends="struts-default" namespace="/">
        <action name="FileUploadAction" class="fileupload.FileUpload">
            <!-- 文件上传有拦截器我们自己定义下,在defaultStrack中的一些定义好的元素会别创建我们可以对他们属性设置 -->
            <interceptor-ref name="defaultStack">
                <param name="fileUpload.allowedExtension">txt,jpg,jar</param>
                <param name="fileUpload.maximumSize">2097152</param>
            </interceptor-ref>
            <result name="success">/e/success.jsp</result>
            <result name="input">/e/error.jsp</result>
        </action>
        
        <action name="down_*" class="fileupload.DownAction" method="{1}">
            <!-- 列表显示 -->
            <result name="list">e/list.jsp</result>
            <!-- 下载操作 -->
            <result name="download" type="stream">
               <!-- 运行下载的文件的类型:指定为所有的二进制文件类型 -->
               <param name="contentType">application/octet-stream</param>
               
               <!-- 对应的是Action中属性: 返回流的属性【其实就是getAttrInputStream()】 -->
               <param name="inputName">attrInputStream</param>
               
               <!-- 下载头,包括:浏览器显示的文件名 -->
               <param name="contentDisposition">attachment;filename=${downFileName}</param>
             
                 <!-- 缓冲区大小设置 -->
               <param name="bufferSize">1024</param>
            </result>
        </action>
    </package> 
    
</struts>
2.FileUpload.java代码
package fileupload;

import java.io.File;

import org.apache.commons.io.FileUtils;
import org.apache.struts2.ServletActionContext;

import com.opensymphony.xwork2.ActionSupport;

/**
 * 文件上传
 * @author Administrator
 *
 */
public class FileUpload extends ActionSupport{
    /**
     * 
     */
    private static final long serialVersionUID = 1L;
    private File file1;
    private String file1FileName;
    //上面的代码是通过拦截器对数据自动封装到这三个属性中的
    
    public void setFile1(File file1) {
        this.file1 = file1;
    }

    public void setFile1FileName(String file1FileName) {
        this.file1FileName = file1FileName;
    }

    @Override
    public String execute() throws Exception {
        String path=ServletActionContext.getServletContext().getRealPath("/upload");
        File destDir=new File(path,file1FileName);
        FileUtils.copyDirectory(file1, destDir);
        System.out.println("上传成功");
        return SUCCESS;
    }
}
3.提交上传的表单jsp
<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
    <form action="/structs/FileUploadAction" method="post" enctype="multipart/form-data">
        文件:<input type="file" name="file1"/>
        <input type="submit" value="上传"/>
    </form>
</body>
</html>
View Code
下载
1.DownAction.java
package fileupload;

import java.io.File;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.util.Map;

import org.apache.struts2.ServletActionContext;

import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionSupport;

public class DownAction extends ActionSupport{

    /**
     * 
     */
    private static final long serialVersionUID = 1L;
    /************显示所有需要下载的文件*************/
    public String list(){
        String path=ServletActionContext.getServletContext().getRealPath("/upload");
        File file=new File(path);
        String[] fileNames =  file.list();//得到所有下载的文件名
        //保存
        ActionContext ac=ServletActionContext.getContext();
        Map<String,Object> request=(Map<String, Object>) ac.get("request");
        request.put("fileNames", fileNames);
        return "list";
    }
    
    /***************2.文件下载****************/
    /*1。获得要下载的文件名字*/
    private String fileName;
    public void setFileName(String fileName){
        //这里提交时get提交存在乱码问题需要解决下
        try {
            fileName=new String(fileName.getBytes("ISO8859-1"),"UTF-8");
        } catch (UnsupportedEncodingException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        this.fileName=fileName;
    }
    /*2.下载提交业务*/
    public String down(){
        return "download";
    }
    /*3.返回文件流的方法*/
    public InputStream getAttrInputStream(){
        return ServletActionContext.getServletContext().getResourceAsStream("/upload"+fileName);
    }
    
    /*4.下载显示的文件名*/
    public String getDownFileName(){
        //需要中文编码
        try {
            fileName=URLEncoder.encode(fileName,"UTF-8");
        } catch (UnsupportedEncodingException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return fileName;
    }
    
}
2.显示所有下载文件的list.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
    <base href="<%=basePath%>">
    
    <title>下载列表</title>
    <meta http-equiv="pragma" content="no-cache">
    <meta http-equiv="cache-control" content="no-cache">
    <meta http-equiv="expires" content="0">    
  </head>
  
  <body>
      list列表
      <table border="1" align="center" >
          <tr>
              <td>编号</td>
              <td>文件名</td>
              <td>操作</td>
          </tr>
          <%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
          <c:forEach items="${fileNames }" varStatus="vs" var="fileName">
              <tr>
                  <td>${vs.count}</td>
                  <td>${fileName}</td>
                  <td>
                      <!-- 创建一个url -->
                      <c:url var="url" value="down_down">
                          <c:param name="fileName" value="${fileName }"></c:param>
                      </c:url>
                      <a href="${url }">下载</a>
                  </td>
              </tr>
        </c:forEach>
      </table>
  </body>
</html>

3。访问方式:先访问list,再下载
View Code

6.拦截器

  a) 想出拦截器原因

    用户想要给action什么功能的时候可以通过拦截器自由组装,基于组件的设计

  b) 知识点:拦截器在struts-default中定义了32种拦截器,18中默认拦截器

    拦截器栈:组合多个拦截器,默认使用strut-default18个默认拦截器defaultStack

    一旦用户指定哪个拦截器,默认拦截器就不起作用了

  c) 拦截器的配置

    在struts-default中定义所有的拦截器其中默认拦截器

<interceptor-stack name="defaultStack">
                <interceptor-ref name="exception"/>
                <interceptor-ref name="alias"/>
                <interceptor-ref name="servletConfig"/>
                <interceptor-ref name="i18n"/>
                <interceptor-ref name="prepare"/>
                <interceptor-ref name="chain"/>
                <interceptor-ref name="scopedModelDriven"/>
                <interceptor-ref name="modelDriven"/>
                <interceptor-ref name="fileUpload"/>
                <interceptor-ref name="checkbox"/>
                <interceptor-ref name="multiselect"/>
                <interceptor-ref name="staticParams"/>
                <interceptor-ref name="actionMappingParams"/>
                <interceptor-ref name="params">
                    <param name="excludeParams">dojo..*,^struts..*,^session..*,^request..*,^application..*,^servlet(Request|Response)..*,parameters...*</param>
                </interceptor-ref>
                <interceptor-ref name="conversionError"/>
                <interceptor-ref name="validation">
                    <param name="excludeMethods">input,back,cancel,browse</param>
                </interceptor-ref>
                <interceptor-ref name="workflow">
                    <param name="excludeMethods">input,back,cancel,browse</param>
                </interceptor-ref>
                <interceptor-ref name="debugging"/>
            </interceptor-stack>
View Code

  自己在struts中定义拦截器

    1.定义拦截器和拦截器栈

    1.1定义拦截器

      <interceptors name=”” class=””></interceptors>

    1.2定义拦截器栈

      <interceptors-stack name=””>

    引用上面的或其他的拦截器

      </interceptors-stack>

    2.默认执行拦截器(栈)

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

  e) 拦截器核心api

    Interceptor接口

    AbstractInterceptor 拦截器默认实现的抽象类,一般自定义开发继承它就行了

    ActionInvacation 拦截器的执行状态,调用下一个拦截器或者action

  f) 拦截器和过滤器的区别

  g) 拦截器的生命周期

    服务器启动-》过滤器创建-》初始化init()创建所有拦截器对象-》客户端访问-》创建action实例-》拦截器interceptor方法拦截-》下一个拦截-》。。。》到达action执行execute()->返回给用户结果

  h) 自定义拦截器例子

1.写一个拦截器类

package interceptor;

import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.interceptor.Interceptor;

/**
 * 拦截器定义
 * @author Administrator
 *
 */
public class HelloInterceptor implements Interceptor{

    /**
     * 
     */
    private static final long serialVersionUID = 1L;

    @Override
    public void destroy() {
        // TODO Auto-generated method stub
        System.out.println("销毁。。。。。");
    }

    @Override
    public void init() {
        // TODO Auto-generated method stub
        System.out.println("自定义拦截器初始化成功");
    }
    /**
     * 拦截器业务处理:在访问action时执行,在excute之前执行
     */
    @Override
    public String intercept(ActionInvocation arg0) throws Exception {
        System.out.println("拦截器开始执行");
        //执行业务逻辑
        //执行下一个拦截器
        String aa=arg0.invoke();
        //拦截器结束
        System.out.println("拦截器结束");
        return aa;
    }
    
}
View Code

  2.配置struts配置拦截器分3

     a) 定义自定义拦截器<interceptor>

    b) 定义我们的拦截器栈<interceptor-stack>:其中引用默认拦截器栈要放在第一位

    c) 执行拦截器:<default-interceptor-ref>

<!-- 拦截器配置 -->
        <interceptors>
            <interceptor name="helloInterceptor" class="interceptor.HelloInterceptor"></interceptor>
            <interceptor-stack name="helloStack">
                <!-- 引用默认栈 (一定要放到第一行)-->
                <interceptor-ref name="defaultStack"></interceptor-ref>
                <!-- 引用自定义拦截器 -->
                <interceptor-ref name="helloInterceptor"></interceptor-ref>
            </interceptor-stack>
        </interceptors>
        <!-- 执行拦截器 -->
        <default-interceptor-ref name="helloStack"></default-interceptor-ref>
        <action name="hello" class="interceptor.HelloAction">
            <result name="success">/index.jsp</result>
        </action>
View Code

7.国际化

  a) Serlvelt中的国际化

  1. 写资源文件

    基础名.properties【默认的语言环境设置】

    基础名_语言简称_国家简称.properties

  1. 读取资源文件再使用

    程序:ResourceBundle

    Jsp: jstl提供的格式化与国际化标签库。

  b) Struts中的国际化

    1.   写资源文件(同servlet
    2.   读取资源文件再使用

        程序:ResourceBundle

        Jsp:  1jstl表亲啊  (同servlet

            2struts标签获取资源文件内容

  c) 注意

    还可以在页面加载

      <s:i18n name="cn.itcast.config.msg">

        <s:text>  标签必须放到标签体中。

      </s:i18n>

  d) 区别

    Struts2加载资源文件更加简单!通过常量加载即可!再在jsp页面直接使用

à1.  写资源文件

Msg.properties   默认的语言环境; 找不到配置就找它

Msg_en_US.properties  美国

-à2.  加载

<constant name="struts.custom.i18n.resources" value="cn.itcast.config.msg"></constant>

à3. 使用: 标签name值直接写配置文件中的key

<s:text name="title"></s:text>

8.Ognl表达式语言与Struts标签

  深刻理解Ognl,Struct2传输模式

http://blog.sina.com.cn/s/blog_7ffb8dd5010141pd.html

1.ActionContext装饰OgnlContext
2.OgnlValueStack值栈对象包含了OgnlContext,root对象
3.OgnlContext中包含了所有域对象,全局属性,action对象等等
4.Struts2数据传输DataTransfer的核心对象是OgnlValueStack、 OgnlContext

   这里主要需要知道的是“值栈对象”生成过程和内部成员,调用方式

  a) Ognl表达式:Object Graphic Navigation Language(对象图导航语言)的缩写是一个开源项 

  b) 优势:

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

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

    1.   支持赋值操作和表达式串联,如price=100, discount=0.8,calculatePrice(),这个表达式会返回80
    2.   访问OGNL上下文(OGNL context)和ActionContext
    3.   操作集合对象

  c) OgnlContext对象

    是Ognl表达式的核心

    访问Ognl表达式中值Ognl.getValue(ognl表达式,ActionContext ac,ognl的root)

    例子:主要看我们对ActionContext上下文对象如何处理,ac中直接添加键值对,创建ognl表达式时要加#,ac添加root时,创建表达式就不需要了

       当我们调用工具类的一个方法时,创建ognl表达式:Ognl.parseExpression("@[类名]@[方法调用]")如:@Math@floor(10.9)

1. Ognl表达式语言语言取值,取非根元素的值,必须用#号

public void testOgnl() throws Exception {
        // 创建一个Ognl上下文对象
        OgnlContext context = new OgnlContext();
        // 放入数据
        User user = new User();
        user.setId(100);
        user.setName("Jack");
        // 【往非根元素放入数据, 取值的时候表达式要用"#"】
        context.put("user", user);
        
        // 获取数据(map)
        // 先构建一个Ognl表达式, 再解析表达式
        Object ognl = Ognl.parseExpression("#user.name");
        Object value = Ognl.getValue(ognl, context, context.getRoot());
        System.out.println(value);
    }
View Code

2. Ognl表达式语言语言取值,取根元素的值,不用带#号

public void testOgn2() throws Exception {
        // 创建一个Ognl上下文对象
        OgnlContext context = new OgnlContext();
        // 放入数据
        User user = new User();
        user.setId(100);
        user.setName("Jack");
        // 【往根元素放入数据】
        context.setRoot(user);
        
        // 获取数据(map)
        // 先构建一个Ognl表达式, 再解析表达式
        Object ognl = Ognl.parseExpression("address.province");
        Object value = Ognl.getValue(ognl, context, context.getRoot());
        
        System.out.println(value);
    }
View Code

3.Ognl对 静态方法调用的支持

public void testOgn3() throws Exception {
        // 创建一个Ognl上下文对象
        OgnlContext context = new OgnlContext();
        
        // Ognl表单式语言,调用类的静态方法
        //Object ognl = Ognl.parseExpression("@Math@floor(10.9)");
        // 由于Math类在开发中比较常用,所以也可以这样写
        Object ognl = Ognl.parseExpression("@@floor(10.9)");
        Object value = Ognl.getValue(ognl, context, context.getRoot());
        System.out.println(value);
    }
View Code

 4.Ognl运用在jsp页面:数据存到request中后自然就存到OgnlContext中了,在jsp页面通过#request.list就能访问到OgnlContext中的list对象和里面的值,OgnlContext就相当于一个容器,存放了所有的域对象和自己存进去的键值对

map迭代
<s:iterator var="en" value="#request.map" status="st">
              <tr>
                  <td><s:property value="#en.key"/></td>
                  <td><s:property value="#en.value.name"/></td>
              </tr>
          </s:iterator>
View Code

不是根Obnl的根对象都是用#开头才能访问到

  d) ValueStack对象

    用户访问action时,创建action对象,之后再将很多信息,action对象、全局属性、域对象、等等存入值栈对象中,之后值栈对象就有了很多内容,我们取值都可以从这里取

    值栈对象,是struts数据存储中心,或者说是中转站

    访问action时,创建了action对象、值栈对象、ActionContext对象;将action对象放入值栈对象,再将值栈对象存入request中传到jsp

    ActionContext类装饰OgnlContext类

    获取值栈对象2种方式:

        1.因为action对象放入值栈对象,值栈对象又存入request对象,所有想要获得ValueStack对象可以从request对象中获得

          ValueStack vs1 = (ValueStack) request.getAttribute("struts.valueStack");

        2.直接从ActionContext对中获得

          ac..getValueStack();

  e) Struts标签

    Struct标签就是用了Ognl表达式语言

 9.验证

代码验证

package validation;

import com.opensymphony.xwork2.ActionSupport;

/**
 * 验证需要实现接口或者继承ActionSupport
 * @author Administrator
 *
 */
public class UserAction extends ActionSupport{

    private static final long serialVersionUID = 1L;
    //封装数据的请求
    private User user;
    public User getUser() {
        return user;
    }
    public void setUser(User user) {
        this.user = user;
    }
    //重写数据验证的方法
    /*public void validateRegister() {
        if(user.getUserName()==null||"".equals(user.getUserName())){
            //保存错误信息
            super.addFieldError("userName","用户名必须重写");
        }
        // 密码
        if (user.getPwd() == null || "".equals(user.getPwd())) {
            super.addFieldError("pwd", "密码必填");
        }
    }*/
    //业务方法
    public String register(){
        System.out.println(user);
        System.out.println(1);
        return SUCCESS;
    }
    //列表展示
    public String list(){
        return SUCCESS;
    }
}
View Code

xml验证

<?xml version="1.0" encoding="UTF-8"?>
 <!DOCTYPE validators PUBLIC
          "-//Apache Struts//XWork Validator 1.0.3//EN"
          "http://struts.apache.org/dtds/xwork-validator-1.0.3.dtd">
 <validators>
     <!-- 验证每一个字段用field表示 -->
    <field name="user.userName">
        <field-validator type="requiredstring">
            <!-- 验证失败的错误提示信息 -->
            <message>用户名不能为空</message>
        </field-validator>
    </field> 
    
    <!-- 密码不能为空 -->
    <field name="user.pwd">
        <!-- 非空 -->
        <field-validator type="requiredstring">
            <message>密码不能为空!</message>
        </field-validator>
        <!-- 长度 -->
        <field-validator type="stringlength">
            <param name="minLength">6</param>
            <param name="maxLength">8</param>
            <message>密码必须为6-8位!</message>    
        </field-validator>
    </field>
    
    <!-- 验证日期 -->
    <field name="user.birth">
        <field-validator type="date">
            <message>日期格式不对!</message>
        </field-validator>
    </field>
    <!-- 验证emil -->
    <field name="user.emil">
        <field-validator type="email">
            <message>邮件格式不对</message>

        </field-validator>
    </field>
 </validators>
View Code

  1)验证原理

    通过拦截器验证

<interceptor name="validation" class="org.apache.struts2.interceptor.validation.AnnotationValidationInterceptor"/>

  2)配置验证步骤

    写一个类继承ActionSuppert或者实现Validate接口重写validate()方法即可

    要注意的是:想要验证指定的方法只需只需验证名称规则:validate+要验证的方法名(public void validateRegister())

  3)验证action的方法

    1)代码验证

      重写验证方法,注意命名规则可以指定特定的方法验证

      validate+要验证的方法名

      public void validateRegister() {

          只会验证当前actionregister方法!

    2)XML方式验证

      1.将错误信息显示在jsp页面:<s:fielderror fieldName="user.userName"></s:fielderror>

      2.代码验证缺点:设计很多重复的验证逻辑!例如:非空验证、数值验证、email、日期等。

      3.Struts对于常用的验证进行了封装

        Struts提供的所有的验证器:xwork-core-2.3.4.1.jar/com.opensymphony.xwork2.validator.validators/default.xml

      都在这里了

<!-- START SNIPPET: validators-default -->
<validators>
    <validator name="required" class="com.opensymphony.xwork2.validator.validators.RequiredFieldValidator"/>
    <validator name="requiredstring" class="com.opensymphony.xwork2.validator.validators.RequiredStringValidator"/>
    <validator name="int" class="com.opensymphony.xwork2.validator.validators.IntRangeFieldValidator"/>
    <validator name="long" class="com.opensymphony.xwork2.validator.validators.LongRangeFieldValidator"/>
    <validator name="short" class="com.opensymphony.xwork2.validator.validators.ShortRangeFieldValidator"/>
    <validator name="double" class="com.opensymphony.xwork2.validator.validators.DoubleRangeFieldValidator"/>
    <validator name="date" class="com.opensymphony.xwork2.validator.validators.DateRangeFieldValidator"/>
    <validator name="expression" class="com.opensymphony.xwork2.validator.validators.ExpressionValidator"/>
    <validator name="fieldexpression" class="com.opensymphony.xwork2.validator.validators.FieldExpressionValidator"/>
    <validator name="email" class="com.opensymphony.xwork2.validator.validators.EmailValidator"/>
    <validator name="url" class="com.opensymphony.xwork2.validator.validators.URLValidator"/>
    <validator name="visitor" class="com.opensymphony.xwork2.validator.validators.VisitorFieldValidator"/>
    <validator name="conversion" class="com.opensymphony.xwork2.validator.validators.ConversionErrorFieldValidator"/>
    <validator name="stringlength" class="com.opensymphony.xwork2.validator.validators.StringLengthFieldValidator"/>
    <validator name="regex" class="com.opensymphony.xwork2.validator.validators.RegexFieldValidator"/>
    <validator name="conditionalvisitor" class="com.opensymphony.xwork2.validator.validators.ConditionalVisitorFieldValidator"/>
</validators>
<!--  END SNIPPET: validators-default -->
View Code

      XML文件名称语法:

          指定的是所有验证:ActionClassName-validation.xml;

          指定特定的方法命名规则:ActionClassName-ActionName-validation.xml

        注意:此XML需要与当前要验证的acton同在一个目录

        举例:UserAction-validation.xml/UserAction-register-validation.xml

例如:xml中的规则在dtd文件在xwork-core-2.3.4.1.jar下慢慢找吧dtd文件

<?xml version="1.0" encoding="UTF-8"?>
 <!DOCTYPE validators PUBLIC
          "-//Apache Struts//XWork Validator 1.0.3//EN"
          "http://struts.apache.org/dtds/xwork-validator-1.0.3.dtd">
 <validators>
     <!-- 验证每一个字段用field表示 -->
    <field name="user.userName">
        <field-validator type="requiredstring">
            <!-- 验证失败的错误提示信息 -->
            <message>用户名不能为空</message>
        </field-validator>
    </field> 
    
    <!-- 密码不能为空 -->
    <field name="user.pwd">
        <!-- 非空 -->
        <field-validator type="requiredstring">
            <message>密码不能为空!</message>
        </field-validator>
        <!-- 长度 -->
        <field-validator type="stringlength">
            <param name="minLength">6</param>
            <param name="maxLength">8</param>
            <message>密码必须为6-8位!</message>    
        </field-validator>
    </field>
    
    <!-- 验证日期 -->
    <field name="user.birth">
        <field-validator type="date">
            <message>日期格式不对!</message>
        </field-validator>
    </field>
    <!-- 验证emil -->
    <field name="user.emil">
        <field-validator type="email">
            <message>邮件格式不对</message>

        </field-validator>
    </field>
 </validators>
View Code

      3)验证总结

        代码:

              重写validate()   ,  验证action所有方法

             Validate方法名(),  验证指定“方法名”的方法

        Xml

            验证所有方法: ActionClassName-validation.xml

            验证指定方法: ActionClassName-actionName-validation.xml

        代码验证,

            比较灵活,可以满足所有的需求.

            比较繁琐,要写重复的验证判断逻辑!

            适合: 表单字段较少的情况用!

        XML验证:

            通用,但不够灵活; 可以验证特定简单的业务。

            适合: 验证表单字段较多,可以大大简化代码!

              (配置文件过多)

    4)显示错误信息:

1.方式一:
<%@taglib uri="/struts-tags" prefix="s" %>
      <!-- 显示的是struts在运行时期产生的所有错误 -->
      <s:fielderror></s:fielderror>
2.方式二:
<!-- 修改struts标签默认的样式: 不让换行 --> 这样弄因为显示错写信息时会自动生成ui,ul标签

  <style type="text/css">

    ul{display: inline;}

    ul li{display: inline;color: red;}

  </style>

  3.修改标签定义的模板

    找到fielderror标签定义的模板文件:Struts-core.jar	emplatesimple fielderror.ftl

    把修改后的文件放到src/ template/ simple/ fielderror.ftl

    这样样式就修改好了 
View Code

10.struts简单的UI标签 

<!-- 服务器标签 : 最终别解析为html标签-->
      <s:form action="/user_login" method="post" name="frmLogin" id="frmLogin" theme="simple">
          
          用户名:<s:textfield name="user.name"></s:textfield>
          密码:<s:textfield name="user.pwd"></s:textfield>
          <s:submit value="登陆"></s:submit>
      </s:form>

<!-- 修改主题 (当前项目所有的标签都用此主题)-->

<constant 

name="struts.ui.theme" value="simple"></constant>

 11.Struts中几种特殊符号(jsp页面)

  #:获取非根元素值、map集合

  $:配置文件取值

  %:提供一个ognl表示的运行环境

例子:

<body>
       <br/>获取request域数据<br/>
       <!-- property 标签是对象类型的标签,默认支持ognl表达式, 会从根元素去China名称对应的值 -->
       <s:property value="China"/>        <br/>
       <!-- 如果直接赋值,需要用单引号 -->
       <s:property value="'China'"/>        <br/>
       <s:property value="%{#request.cn}"/>        <br/>
       
       <!-- 值类型的标签,value值默认就是值类型,不支持ognl表达式 -->
       国家:<s:textfield name="txtCountry" value="%{#request.cn}"></s:textfield>
  </body>
View Code

12.Struts中常用的几个技术

  1)数据回显

    必须使用struts标签:

    进入修改页面时-》将数据信息存入request域Map中,或者存入栈值对中,在jsp页面,通过struts标签就能获得obnl标签值

    两种数据存入方式

      1.通过request域Map

      2.通过ValueStack存入头对象,能直接访问

例子:

Action中:
// 进入修改页面
    public String viewUpdate() {
        // 模拟一个对象(先获取一个id,再根据id调用service查询,把查到的结果保存到域)
        User userInfo = new User();
        userInfo.setUserName("Jack");
        userInfo.setEmail("yuanjie@itcast.cn");
        
        ActionContext ac = ActionContext.getContext();
//        Map<String,Object> request = (Map<String, Object>) ac.get("request");
//        request.put("userInfo", userInfo);
        
        /************* 数据回显***************/
        // 获取值栈
        ValueStack vs = ac.getValueStack();
        vs.pop();// 移除栈顶元素
        vs.push(userInfo);  // 入栈
        
        
        // 进入修改页面
        return "viewUpdate";
    }
    
JSP页面:
<body>
      <%@taglib uri="/struts-tags" prefix="s" %>
      
      
      <br/>
      <!-- 在页面文本框内,显示要修改记录的数据 -->
      
      <!-- 手动通过value设置显示的值 
      <s:form action="#">
          
          用户名: <s:textfield name="user.userName" value="%{#request.userInfo.userName}"></s:textfield>   <br/>
          
          邮箱: <s:textfield name="user.email" value="%{#request.userInfo.email}"></s:textfield>     <br/>
      </s:form>
      -->
      
      <!-- 数据回显技术:s:textfield会自动查找根元素数据(Ognl表达式语言取值)  -->
      <s:form action="#">
          
          用户名: <s:textfield name="userName"></s:textfield>   <br/>
          
          邮箱: <s:textfield name="email"></s:textfield>     <br/>
      </s:form>
      
      <s:debug></s:debug>
  </body>
View Code

  2)模型驱动,属性驱动

    模型驱动就是直接将对象封装,属性驱动就是将属性赋值给对象的属性

    <interceptor name="modelDriven" class="com.opensymphony.xwork2.interceptor.ModelDrivenInterceptor"/>

@Override
    public String intercept(ActionInvocation invocation) throws Exception {
        Object action = invocation.getAction();

        if (action instanceof ModelDriven) {
            ModelDriven modelDriven = (ModelDriven) action;
            ValueStack stack = invocation.getStack();
            Object model = modelDriven.getModel();
            if (model !=  null) {
                stack.push(model);
            }
            if (refreshModelBeforeResult) {
                invocation.addPreResultListener(new RefreshModelBeforeResult(modelDriven, model));
            }
        }
        return invocation.invoke();
    }
View Code

    prams拦截器,可以把请求数据自动填充的action的属性中

    就是数据封装的关键拦截器

  4.表单重复提交拦截器

    <interceptor name="token" class="org.apache.struts2.interceptor.TokenInterceptor"/>

Struts2知识进阶

1.继承了ActionSupport的类,想要给前台页面发送信息处了可以通过域对象发送;还可以通过以下三种方式提交信息

  this.addActionMessage("message");
  this.addActionError(anErrorMessage);
  this.addFieldError(fieldName, errorMessage)

  之后在前台界面使用<%@ taglib uri="/struts-tags" prefix="s"%>

              <s:actionmessage/>;<s:actionerror/>来显示数据信息

原文地址:https://www.cnblogs.com/xiaoping1993/p/6889563.html