Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 261|回复: 0

[struts学习]在Struts 2_0中实现表单数据校验

[复制链接]
  • TA的每日心情
    开心
    2021-3-12 23:18
  • 签到天数: 2 天

    [LV.1]初来乍到

    发表于 2014-10-10 23:52:29 | 显示全部楼层 |阅读模式
    All Input Is Evil!
    -Writing secure code  
       
       
      
          在写前几篇文章的时候,有些朋友建议我的写一篇关于表单数据校验的文章。 正如文章的开头所引用的《Writing Secure Code》的名言:“所有的输入都是罪恶的”,所以我们应该对所有的外部输入进行校验。而表单是应用程序最简单的入口,对其传进来的数据,我们必须进行校验。 转换与校验(Conversion & Validation) 其实上篇文章,我本来是打算写表单数据校验的内容,但是经过再三思考后,还是决定先写Struts 2.0转换器的内容。原因是我认为转换是校验的基础,只有在数据被正确地转换成其对应的类型后,我们才可以对其取值范围进行校验。看个例子相信大家可以更清楚。现在我们就来改造一下《转换器(Converter)――Struts 2.0中的魔术师》的第一个例子。 首先,从Action开始,修改后的代码如下:


    package
    tutorial;


    import
    java.util.Locale;


    import
    com.opensymphony.xwork2.ActionSupport;

    import
    com.opensymphony.xwork2.util.LocalizedTextUtil;


    public

    class
    HelloWorld
    extends
    ActionSupport

    {
         private String msg;
         private Locale loc = Locale.US;
       
         public String getMsg() {
             return msg;        
        }
       
         public Locale getLoc() {
             return loc;
        }
       
         public void setLoc(Locale loc) {
             this .loc = loc;
        }
       
        @Override
         public void validate() {
            System.out.println( " Calling validate() " );
             if ( ! (loc.equals(Locale.US) || loc.equals(Locale.CHINA))) {
                        addFieldError( " loc " , getText( " validation.loc " ));
            }
        }
            
         public void validateExecute() {
            System.out.println( " Calling validateExecute() by reflection " );
        }
       
        @Override
         public String execute() {
            System.out.println( " Calling execute() " );
             // LocalizedTextUtil是Struts 2.0中国际化的工具类,<s:text>标志就是通过调用它实现国际化的
                msg = LocalizedTextUtil.findDefaultText( " HelloWorld " , loc);
             return SUCCESS;
        }
    }
    然后,修改Struts.xml中Action的定义指明输入地址:

    <
    action
    name
    ="HelloWorld"
    class
    ="tutorial.HelloWorld"
    >

         
    <
    result
    >
    /HelloWorld.jsp
    </
    result
    >

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


    </
    action
    >
    接着,在HelloWorld.jsp中加入错误提示:

    <%
    @ page  contentType
    =
    "
    text/HTML; charset=UTF-8
    "
    %>


    <%
    @taglib prefix
    =
    "
    s
    "
    uri
    =
    "
    /struts-tags
    "

    %>


    <
    html
    >


    <
    head
    >

         
    <
    title
    >
    Hello World
    </
    title
    >


    </
    head
    >


    <
    body
    >

         
    <
    div
    style
    ="color:red;"
    >

             
    <
    s:fielderror
    />

         
    </
    div
    >

         
    <
    s:form
    action
    ="HelloWorld"
    theme
    ="simple"
    >
                
             Locale:
    <
    s:textfield
    name
    ="loc"

    />
       
    <
    s:submit
    />

         
    </
    s:form
    >
       
         
    <
    h2
    ><
    s:property
    value
    ="msg"

    /></
    h2
    >


    </
    body
    >


    </
    html
    >
    再修改LocaleConverter.java文件,将内容改为:


    package
    tutorial;


    import
    java.util.Locale;

    import
    java.util.Map;

    import
    java.util.regex.Pattern;


    public

    class
    LocaleConverter
    extends
    ognl.DefaultTypeConverter

    {
        @Override
         public Object convertValue(Map context, Object value, Class toType) {
             if (toType == Locale. class ) {            
                System.out.println( " Converting String to Locale " );
                String locale = ((String[]) value)[ 0 ];
                 return new Locale(locale.substring( 0 , 2 ), locale.substring( 3 ));
            } else if (toType == String. class ) {
                System.out.println( " Converting Locale to String " );
                Locale locale = (Locale) value;
                 return locale.toString();
            }
             return null ;
        }
    }
    之后,修改国际化资源文件,内容为:

    HelloWorld
    =
    你好,世界!
    invalid.fieldvalue.loc
    =
    Locale必须为
    "
    xx_XX
    "
    的格式
    validation.loc
    =
    区域必须为中国或美国
    globalMessages_zh_CN.properties



    HelloWorld
    =
    Hello World!
    invalid.fieldvalue.loc
    =
    Locale must like  
    "
    xx_XX
    "

    validation.loc
    =
    Locale must be China or USA
    globalMessages_en_US.properties 发布运行应用程序,在浏览器中键入http://localhost:8080/Struts2_Validation/HelloWorld.action,在Locale中输入zh_CN,按“Submit”提交,效果如上篇文章所示。而在服务器控制台有如下输出:

    Converting String to Locale...
    Calling validateExecute() by reflection...
    Calling validate()...
    Calling execute()...
    Converting Locale to String...

    上述的输出说明了Struts 2.0的数据校验工作方式,它需要经过下面几个步骤:

    通过转换器将请求参数转换成相应的Bean属性;
    判断转换过程是否出现异常。如果有,则将其保存到ActionContext中,conversionError拦截器再封装为fieldError;如果没有,进行下一步;
    通过反射(Reflection)来调用validateXxx()方法(其中,Xxx表示Action的方法名);
    调用validate()方法;
    如果经过上述步骤没有出现fieldError,则调用Action方法;如果有,则会跳过Action方法,通过国际化将fieldError输出到页面。  
    不喜欢看文字的朋友,可以参考下面的图1。
    图1 校验顺序图  看到这里可能大家会疑问:“这么多地方可以校验表单数据,到底我应该在那里做呢?”有选择是好事,但抉择的过程往往是痛苦的,往往让人不知所措。如果大家参照以下几点建议,相信会比较容易地做出正确的抉择。

    如果需要转换的数据,通常做法在转换的时候做格式的校验,在Action中的校验方法中校验取值。假如用户填错了格式,我们可以通过在资源文件配置invalid.fieldvalue.xxx(xxx为属性名)来提示用户正确的格式,不同的阶段出错显示不同的信息。具体做法请参考上面的例子;
    至于用validate()还是validateXxx(),我推荐使用validate()。原因是validateXxx()使用了反射,相对来说性能稍差,而validate()则是通过接口com.opensymphony.xwork2.Validateable调用。当然如果你的表单数据取值是取决于特定Action方法,则应该使用validateXxx()。  
    在运行上面的例子时,在Locale中输入zh并提交时出现图2所示页面。
    图2 错误格式  在Locale中输入de_DE时,出现如图3所示页面。
    图3 取值错误  使用Struts 2.0的校验框架 上一节的内容都是关于如何编程实现校验,这部分工作大都是单调的重复。更多情况下,我们使用Struts 2.0的校验框架,通过配置实现一些常见的校验。 我学习编程有个习惯――喜欢先看输出结果,再看代码实现。这样学的好处是先看结果可以刺激学习的激情,也可以在看代码前自已思考一下如何实现,然后带着问题去看代码,那就清晰多了。因此下面我们先来做演示。 首先,在tutorial包下新建ValidationAction.java,代码如下:


    package
    tutorial;


    import
    com.opensymphony.xwork2.ActionSupport;


    public

    class
    ValidationAction
    extends
    ActionSupport

    {
         private String reqiuredString;

         public String getReqiuredString() {
             return reqiuredString;
        }

         public void setReqiuredString(String reqiuredString) {
             this .reqiuredString = reqiuredString;
        }
       
        @Override
         public String execute() {
             return SUCCESS;
        }   
    }
    然后,配置上述所建的Ation,代码片段如下:

    <
    action
    name
    ="ValidationAction"
    class
    ="tutorial.ValidationAction"
    >

         
    <
    result
    >
    /Output.jsp
    </
    result
    >

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


    </
    action
    >
    接着,创建Input.jsp和Output.jsp,内容分别如下:

    <%
    @ page  contentType
    =
    "
    text/html; charset=UTF-8
    "
    %>


    <%
    @taglib prefix
    =
    "
    s
    "
    uri
    =
    "
    /struts-tags
    "

    %>


    <
    html
    >


    <
    head
    >

         
    <
    title
    >
    Hello World
    </
    title
    >

         
    <!--
    此标志的作用是引入Struts 2.0的常用的Javascript和CSS
    -->

         
    <
    s:head
    />


    </
    head
    >


    <
    body
    >

         
    <
    s:form
    action
    ="ValidationAction"
    >
                
             
    <
    s:textfield
    name
    ="reqiuredString"
    label
    ="Required String"

    />

             
    <
    s:submit
    />

         
    </
    s:form
    >
       

    </
    body
    >


    </
    html
    >
    Input.jsp



    <%
    @ page  contentType
    =
    "
    text/html; charset=UTF-8
    "
    %>


    <%
    @taglib prefix
    =
    "
    s
    "
    uri
    =
    "
    /struts-tags
    "

    %>


    <
    html
    >


    <
    head
    >

         
    <
    title
    >
    Hello World
    </
    title
    >


    </
    head
    >


    <
    body
    >

         Required String:
    <
    s:property
    value
    ="reqiuredString"
    />
       

    </
    body
    >


    </
    html
    >
    Output.jsp 再接下来,在tutorial包下创建ValidationAction的校验配置文件Xxx-validation.xml(Xxx为Action的类名),在本例中该文件名ValidationAction-validation.xml,内容如下:

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


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

    <
    validators
    >

         
    <
    field
    name
    ="reqiuredString"
    >

             
    <
    field-validator
    type
    ="requiredstring"
    >

                
    <
    message
    >
    This string is required
    </
    message
    >

             
    </
    field-validator
    >

         
    </
    field
    >


    </
    validators
    >
    发布运行应用程序,在地址栏中键入http://localhost:8080/Struts2_Validation/Input.jsp,出现如图4所示页面。
    图4 Input.jsp  直接点击“Submit”提交表单,出现图5所示的页面。
    图5 错误提示  在Required String中随便填点东西,转到Output.jsp页面,如图6所示。
    图6 Output.jsp  通过上面的例子,大家可以看到使用该校验框架十分简单方便。不过,上例还有两点不足:

    还没有国际化错误消息;
    没有实现客户端的校验。  
    当然,要完善以上不足,对于Struts 2.0来说,只是小菜一碟。

    在Xxx-validation.xml文件中的<message>元素中加入key属性;
    在Input.jsp中的<s:form>标志中加入validate="true"属性,就可以在用Javascript在客户端校验数据。  
    下面是具体的实现,首先在国际化资源文件中加入错误消息,然后按照上面说明实现。因为要使用Javascript在客户端显示出错信息,所以在加载Input.jsp页面时,Struts 2.0需要获得国际化的字符串,故我们需要使用Action来访问Input.jsp页面,具体实现请参考《在Struts 2.0中国际化(i18n)您的应用程序》的最后部分。最后发布运行应用程序,直接在页面中点击“Submit”,表单没有被提交并出现错误提示,如图7所示:
    图7 客户端校验  校验框架是通过validation拦截器实现,该拦载被注册到默认的拦截器链中。它在conversionError拦截器之后,在validateXxx()之前被调用。这里又出现了一个选择的问题:到底是应该在action中通过validateXxx()或validate()实现校验,还是使用validation拦截器?绝大多数情况,我建议大家使用校验框架,只有当框架满足不了您的要求才自已编写代码实现。 配置文件查找顺序 在上面的例子中,我们通过创建ValidationAction-validation.xml来配置表单校验。Struts 2.0的校验框架自动会读取该文件,但这样就会引出一个问题――如果我的Action继承其它的Action类,而这两个Action类都需要对表单数据进行校验,那我是否会在子类的配置文件(Xxx-validation.xml)中复制父类的配置吗? 答案是不,因为Struts 2.0的校验框架跟《在Struts 2.0中国际化(i18n)您的应用程序》提到的“资源文件查找顺序”相似,有特定的配置文件查找顺序。不同的是校验框架按照自上而下的顺序在类层次查找配置文件。假设以下条件成立:

    接口 Animal;
    接口 Quadraped 扩展了 Animal;
    类 AnimalImpl 实现了 Animal;
    类 QuadrapedImpl 扩展了 AnimalImpl 实现了 Quadraped;
    类 Dog 扩展了 QuadrapedImpl;  
    如果Dog要被校验,框架方法会查找下面的配置文件(其中别名是Action在struts.xml中定义的别名):

    Animal-validation.xml
    Animal-别名-validation.xml
    AnimalImpl-validation.xml
    AnimalImpl-别名-validation.xml
    Quadraped-validation.xml
    Quadraped-别名-validation.xml
    QuadrapedImpl-validation.xml
    QuadrapedImpl-别名-validation.xml
    Dog-validation.xml
    Dog-别名-validation.xml  
    已有的校验器 Struts 2.0已经为您实现很多常用的校验了,以下在jar的default.xml中的注册的校验器。

    <
    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
    ="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"
    />


    </
    validators
    >




      
       
       
          
        关于每个校验器的具体用法,请参考以下链接:
    http://wiki.javascud.org/display/ww2cndoc/Validation
    该链接中还有一些很有的信息,请大家仔细阅读。
       
       
      
    总结 使用校验框架既可以方便地实现表单数据校验,又能够将校验与Action分离,故我们应该尽可能使用校验框架。在使用校验框架时,请不要忘记通过在资源文件加入invalid.fieldvalue.xxx字符串,显示适合的类型转换出错信息;或者使用conversion校验器。
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-2-26 13:17 , Processed in 0.397462 second(s), 46 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表