java form validation_在Struts 2.0中实现表单数据校验(Validation)

All Input Is Evil!

-Writing secure code

在写前几篇文章的时候,有些朋友建议我的写一篇关于表单数据校验的文章。 正如文章的开头所引用的《Writing Secure Code》的名言:“所有的输入都是罪恶的”,所以我们应该对所有的外部输入进行校验。而表单是应用程序最简单的入口,对其传进来的数据,我们必须进行校验。

转换与校验(Conversion & Validation)

其实上篇文章,我本来是打算写表单数据校验的内容,但是经过再三思考后,还是决定先写Struts 2.0转换器的内容。原因是我认为转换是校验的基础,只有在数据被正确地转换成其对应的类型后,我们才可以对其取值范围进行校验。看个例子相信大家可以更清楚。现在我们就来改造一下《转换器(Converter)——Struts

2.0中的魔术师》的第一个例子。

首先,从Action开始,修改后的代码如下:

4f1150b881333f12a311ae9ef34da474.png

package

tutorial;

4f1150b881333f12a311ae9ef34da474.png

4f1150b881333f12a311ae9ef34da474.png

import

java.util.Locale;

4f1150b881333f12a311ae9ef34da474.png

4f1150b881333f12a311ae9ef34da474.png

import

com.opensymphony.xwork2.ActionSupport;

4f1150b881333f12a311ae9ef34da474.png

import

com.opensymphony.xwork2.util.LocalizedTextUtil;

4f1150b881333f12a311ae9ef34da474.png

1fa987a29c6482f53d401256f96355eb.png

ca75c07623e1b494fee67e8f316fc310.gif

public

class

HelloWorld

extends

ActionSupport

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

private

String msg;

d18c02628675d0a2c816449d98bda930.png

private

Locale loc

=

Locale.US;

d18c02628675d0a2c816449d98bda930.png

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

String getMsg()

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

return

msg;

ecedf933ec37d714bd4c2545da43add2.png

}

d18c02628675d0a2c816449d98bda930.png

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

Locale getLoc()

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

return

loc;

ecedf933ec37d714bd4c2545da43add2.png

}

d18c02628675d0a2c816449d98bda930.png

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

void

setLoc(Locale

loc)

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

this

.loc

=

loc;

ecedf933ec37d714bd4c2545da43add2.png

}

d18c02628675d0a2c816449d98bda930.png

d18c02628675d0a2c816449d98bda930.png

@Override

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

void

validate()

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

System.out.println(

"

Calling validate()

9b8a8a44dd1c74ae49c20a7cd451974e.png

"

);

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

if

(

!

(loc.equals(Locale.US)

||

loc.equals(Locale.CHINA)))

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

addFieldError(

"

loc

"

, getText(

"

validation.loc

"

));

ecedf933ec37d714bd4c2545da43add2.png

}

ecedf933ec37d714bd4c2545da43add2.png

}

d18c02628675d0a2c816449d98bda930.png

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

void

validateExecute()

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

System.out.println(

"

Calling validateExecute() by reflection

9b8a8a44dd1c74ae49c20a7cd451974e.png

"

);

ecedf933ec37d714bd4c2545da43add2.png

}

d18c02628675d0a2c816449d98bda930.png

d18c02628675d0a2c816449d98bda930.png

@Override

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

String execute()

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

System.out.println(

"

Calling execute()

9b8a8a44dd1c74ae49c20a7cd451974e.png

"

);

d18c02628675d0a2c816449d98bda930.png

//

LocalizedTextUtil是Struts 2.0中国际化的工具类,标志就是通过调用它实现国际化的

d18c02628675d0a2c816449d98bda930.png

msg

=

LocalizedTextUtil.findDefaultText(

"

HelloWorld

"

, loc);

d18c02628675d0a2c816449d98bda930.png

return

SUCCESS;

ecedf933ec37d714bd4c2545da43add2.png

}

8f1ba5b45633e9678d1db480c16cae3f.png}

然后,修改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文件,将内容改为:

4f1150b881333f12a311ae9ef34da474.png

package

tutorial;

4f1150b881333f12a311ae9ef34da474.png

4f1150b881333f12a311ae9ef34da474.png

import

java.util.Locale;

4f1150b881333f12a311ae9ef34da474.png

import

java.util.Map;

4f1150b881333f12a311ae9ef34da474.png

import

java.util.regex.Pattern;

4f1150b881333f12a311ae9ef34da474.png

1fa987a29c6482f53d401256f96355eb.png

ca75c07623e1b494fee67e8f316fc310.gif

public

class

LocaleConverter

extends

ognl.DefaultTypeConverter

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

@Override

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

Object convertValue(Map context, Object value, Class toType)

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

if

(toType

==

Locale.

class

)

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

System.out.println(

"

Converting String to Locale

9b8a8a44dd1c74ae49c20a7cd451974e.png

"

);

d18c02628675d0a2c816449d98bda930.png

String locale

=

((String[]) value)[

0

];

d18c02628675d0a2c816449d98bda930.png

return

new

Locale(locale.substring(

0

,

2

), locale.substring(

3

));

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

}

else

if

(toType

==

String.

class

)

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

System.out.println(

"

Converting Locale to String

9b8a8a44dd1c74ae49c20a7cd451974e.png

"

);

d18c02628675d0a2c816449d98bda930.png

Locale locale

=

(Locale) value;

d18c02628675d0a2c816449d98bda930.png

return

locale.toString();

ecedf933ec37d714bd4c2545da43add2.png

}

d18c02628675d0a2c816449d98bda930.png

return

null

;

ecedf933ec37d714bd4c2545da43add2.png

}

8f1ba5b45633e9678d1db480c16cae3f.png}

之后,修改国际化资源文件,内容为:

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。

e1819c1c77749d386ad2ed707b80bdc1.png

图1 校验顺序图

看到这里可能大家会疑问:“这么多地方可以校验表单数据,到底我应该在那里做呢?”有选择是好事,但抉择的过程往往是痛苦的,往往让人不知所措。如果大家参照以下几点建议,相信会比较容易地做出正确的抉择。

如果需要转换的数据,通常做法在转换的时候做格式的校验,在Action中的校验方法中校验取值。假如用户填错了格式,我们可以通过在资源文件配置invalid.fieldvalue.xxx(xxx为属性名)来提示用户正确的格式,不同的阶段出错显示不同的信息。具体做法请参考上面的例子;

至于用validate()还是validateXxx(),我推荐使用validate()。原因是validateXxx()使用了反射,相对来说性能稍差,而validate()则是通过接口com.opensymphony.xwork2.Validateable调用。当然如果你的表单数据取值是取决于特定Action方法,则应该使用validateXxx()。

在运行上面的例子时,在Locale中输入zh并提交时出现图2所示页面。

20918f405e23636788efe3da4aaed172.gif

图2 错误格式

在Locale中输入de_DE时,出现如图3所示页面。

52353e9e30300cd9a065d5363c584c5a.gif

图3 取值错误

使用Struts 2.0的校验框架

上一节的内容都是关于如何编程实现校验,这部分工作大都是单调的重复。更多情况下,我们使用Struts 2.0的校验框架,通过配置实现一些常见的校验。

我学习编程有个习惯——喜欢先看输出结果,再看代码实现。这样学的好处是先看结果可以刺激学习的激情,也可以在看代码前自已思考一下如何实现,然后带着问题去看代码,那就清晰多了。因此下面我们先来做演示。

首先,在tutorial包下新建ValidationAction.java,代码如下:

4f1150b881333f12a311ae9ef34da474.png

package

tutorial;

4f1150b881333f12a311ae9ef34da474.png

4f1150b881333f12a311ae9ef34da474.png

import

com.opensymphony.xwork2.ActionSupport;

4f1150b881333f12a311ae9ef34da474.png

1fa987a29c6482f53d401256f96355eb.png

ca75c07623e1b494fee67e8f316fc310.gif

public

class

ValidationAction

extends

ActionSupport

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

private

String reqiuredString;

d18c02628675d0a2c816449d98bda930.png

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

String getReqiuredString()

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

return

reqiuredString;

ecedf933ec37d714bd4c2545da43add2.png

}

d18c02628675d0a2c816449d98bda930.png

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

void

setReqiuredString(String

reqiuredString)

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

this

.reqiuredString

=

reqiuredString;

ecedf933ec37d714bd4c2545da43add2.png

}

d18c02628675d0a2c816449d98bda930.png

d18c02628675d0a2c816449d98bda930.png

@Override

97e794c86028c5f5b5461ae5ef440a4c.png

3c6cafce68eb941a00f1998f1d3d3aa6.gif

public

String execute()

9b8a8a44dd1c74ae49c20a7cd451974e.png

{

d18c02628675d0a2c816449d98bda930.png

return

SUCCESS;

ecedf933ec37d714bd4c2545da43add2.png

}

8f1ba5b45633e9678d1db480c16cae3f.png}

然后,配置上述所建的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

>

<

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

>

25f8e9c6c7c184aca8a308237920bb8d.gif

图4 Input.jsp

直接点击“Submit”提交表单,出现图5所示的页面。

78257f54c3c0dd7ea5da45f54b994c37.gif

图5 错误提示

在Required String中随便填点东西,转到Output.jsp页面,如图6所示。

1e75c89e1cad428b8fe832155213db73.gif

图6 Output.jsp

通过上面的例子,大家可以看到使用该校验框架十分简单方便。不过,上例还有两点不足:

还没有国际化错误消息;

没有实现客户端的校验。

当然,要完善以上不足,对于Struts 2.0来说,只是小菜一碟。

在Xxx-validation.xml文件中的元素中加入key属性;

在Input.jsp中的

标志中加入validate="true"属性,就可以在用Javascript在客户端校验数据。

下面是具体的实现,首先在国际化资源文件中加入错误消息,然后按照上面说明实现。因为要使用Javascript在客户端显示出错信息,所以在加载Input.jsp页面时,Struts

2.0需要获得国际化的字符串,故我们需要使用Action来访问Input.jsp页面,具体实现请参考《在Struts 2.0中国际化(i18n)您的应用程序》的最后部分。最后发布运行应用程序,直接在页面中点击“Submit”,表单没有被提交并出现错误提示,如图7所示:

ffcda02bc8a9114e1b16210eca0809ec.gif

图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

>

总结

使用校验框架既可以方便地实现表单数据校验,又能够将校验与Action分离,故我们应该尽可能使用校验框架。在使用校验框架时,请不要忘记通过在资源文件加入invalid.fieldvalue.xxx字符串,显示适合的类型转换出错信息;或者使用conversion校验器。

posted on 2006-11-14 13:38 Max 阅读(50951) 评论(118)  编辑  收藏 所属分类: Struts 2.0系列

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值