- 浏览: 116433 次
- 性别:
- 来自: 南京
最新评论
-
yuanfen2014:
学习,学习了
Nginx 301永久重定向 -
yuanfen2014:
[color=blue][/color]
Nginx 301永久重定向 -
lippor:
如何读取啊,怎么都是创建
使用PHP创建 PowerPoint2007 文档 -
huangyunbin:
有很多表的时候,一个一个手动改吗,好麻烦啊,有不有快的方法
mysql导入数据时 USING BTREE 错误解决办法 -
fanyilong_v5:
哥们 第二种 方法 牛逼完了
关于phpcms v9的get标签里面的sql 语句limit无效问题的解决方法
这段时间重新开始拾起JAVA的学习,也是为了以后的工作打算,积累一些经验。
记得开始的时候,学习Struts2中的输入校验,就对Vistor校验器情有独钟。首先,一般我们在提交表单的时候,对表单的各字段就要有校验的过程,实际部署过程中要分 客户端校验 和服务器端校验。 客户端校验使用rapid validation等第三方客户端校验框架就可以实现,而且比Struts2本身自带的客户端校验方式更好,但服务器端校验却可以分为好几种:
1.如果在Action中只有一个处理逻辑的话,可以重写ActionSupport中的validate()方法。
2.如果在Action中存在多个类似于execute的处理逻辑的话,我们可以重写validateXxx()方法(Xxx为不同的处理逻辑名称,例如有validateRegist()方法,就是处理Regist业务逻辑)
3.编写对应于Action的validate 的xml文件。这个应该是最为常用的一种解决方案了,但是有一个问题,Action众多,或者在Action中有多个对应一个或多个POJO的处理逻辑,那么这些xml文件就会随着Action文件和处理逻辑的增加而相应的增加,这显然不是个好的现象。 所以如果只是针对POJO来写相应的validate xml文件,无疑会好很多。 这就是Visitor校验器的功用了。
很多和我一样的初学者可能手边都有一般李刚写的 《Struts 2 权威指南》,里面正好讲了visitor校验器,但是我照着书上讲的步骤重复做了N次都没有正确,于是乎在网上搜索了相关的文章,最后终于发现书上没有讲完整 (P280-P281)
以下我来举个例子:
我建立了一个User的POJO,处理逻辑为UserAction, 测试页面为login.jsp
login.jsp
<%@ page language="java" pageEncoding="UTF-8"%> <%@ taglib prefix="s" uri="/struts-tags"%> <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <title>用户登录页面</title> <style type="text/css"> .errorMessage { font-weight: bold; color: red; } </style> </head> <body> <s:form action="pro_login.do"> <s:textfield name="user.username" label="用户名称" /> <s:password name="user.userpwd" label="登录密码" /> <s:submit value="登 录" /> </s:form> </body> </html>
那么我的Visitor校验器为
UserAction-validation.xml
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE validators PUBLIC "-//OpenSymphony Group//XWork Validator 1.0.2//EN" "http://www.opensymphony.com/xwork/xwork-validator-1.0.2.dtd"> <validators> <field name="user"> <field-validator type="visitor"> <param name="context">userContext</param> <param name="appendPrefix">true</param> <message></message> </field-validator> </field> </validators>
-------------------------------------------------------------------
(注:context参数可以随便设置,只要保持一致就可以了)
-------------------------------------------------------------------
User-userContext-validate.xml
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE validators PUBLIC "-//OpenSymphony Group//XWork Validator 1.0.2//EN" "http://www.opensymphony.com/xwork/xwork-validator-1.0.2.dtd"> <validators> <field name="username"> <field-validator type="requiredstring"> <message>请输入用户名称</message> </field-validator> </field> <field name="userpwd"> <field-validator type="requiredstring"> <message>请输入登录密码</message> </field-validator> </field> </validators>
其中context参数将作为验证User类属性的文件名的一部分,如user属性返回一个User对象,那么用于验证User对象属性的文件名为User-abc-validation.xml。
这个文件要和User.class文件在同一个目录中。也就是要放到你的POJO文件相同的目录路径下,不然的话Visitor校验器就不会起作用。
这个就是李刚老师没有写全的地方,附加个项目截图就更为清楚了。
这样的话,Visitor校验器就起作用了。
评论
<div class="quote_div">
<div class="quote_title">qchong 写道</div>
<div class="quote_div">
<div class="quote_title">jiyanliang 写道</div>
<div class="quote_div">
<div class="quote_title">kjj 写道</div>
<div class="quote_div">password 和repasswrod 可以设置到pojo的属性,然后验证用表达式来做</div>
<br />这个设置岂不是变成了dto了?</div>
<br />干嘛要把repassword设置到POJO的属性?repassword本身只是来确保密码输入正确的,又不是ENTITY的一个属性,传给ACTION用expression验证不就可以了 <br /></div>
<br />传给action 后怎么用exp 验证,还请教!</div>
<pre name="code" class="java">@FieldExpressionValidator(message = "两次密码必须一致", fieldName = "user.userpwd", expression = "user.userpwd == reuserpwd")</pre>
<p> </p>
<p> </p>
这个设置岂不是变成了dto了?
干嘛要把repassword设置到POJO的属性?repassword本身只是来确保密码输入正确的,又不是ENTITY的一个属性,传给ACTION用expression验证不就可以了
传给action 后怎么用exp 验证,还请教!
<div class="quote_div">
<div class="quote_title">cometlj 写道</div>
<div class="quote_div">
<p> 很多和我一样的初学者可能手边都有一般李刚写的 《Struts 2 权威指南》,</p>
</div>
<p><br />首先特别感兴趣这句话。</p>
<p> </p>
<p>其次,该功能我在struts2下用纯annotation方式没有配置成功,struts2的doc关于annotation也是几句带过,关键就是那个context还没搞明白怎么用annotation声明,别告诉我用xml,因为我已经用了annotation,不想写xml。等高人解惑。</p>
<p> </p>
<pre name="code" class="java"> @Validations(
visitorFields = {
@VisitorFieldValidator(message = "Default message", fieldName="user.username", shortCircuit = true, appendPrefix = true),
@VisitorFieldValidator(message = "Default message", fieldName="user.password", shortCircuit = true, appendPrefix = true)
}
)</pre>
<p> </p>
</div>
<p><br />难道你不懂实现 ModelDriven,Preparable?</p>
这个设置岂不是变成了dto了?
干嘛要把repassword设置到POJO的属性?repassword本身只是来确保密码输入正确的,又不是ENTITY的一个属性,传给ACTION用expression验证不就可以了
这个设置岂不是变成了dto了?
第一次听人这么说!不过这样的话,我发现,基于announced 的Hibernian validat 更方便
<div class='quote_div'>password 和repasswrod 可以设置到pojo的属性,然后验证用表达式来做</div>
<p><br/><br/> 多谢 <strong>kjj</strong> 提醒,现在把改进的能够进行确认密码检测的代码放上来,本地测试通过 </p>
<p> </p>
<p>修改后的User的POJO为(<span style='color: #ff0000;'>repass</span>为添加的属性)</p>
<pre name='code' class='java'>package org.lj.domain.hibernate;
import java.util.Date;
/**
* User entity.
*
* @author MyEclipse Persistence Tools
*/
public class User implements java.io.Serializable {
// Fields
private Integer id;
private String username;
private String userpwd;
private String repass;
private Date createdAt;
private Date updatedAt;
// Constructors
/** default constructor */
public User() {
}
/** minimal constructor */
public User(String username, String userpwd, Date createdAt) {
this.username = username;
this.userpwd = userpwd;
this.createdAt = createdAt;
}
/** full constructor */
public User(String username, String userpwd, Date createdAt, Date updatedAt) {
this.username = username;
this.userpwd = userpwd;
this.createdAt = createdAt;
this.updatedAt = updatedAt;
}
// Property accessors
public Integer getId() {
return this.id;
}
public void setId(Integer id) {
this.id = id;
}
public String getUsername() {
return this.username;
}
public void setUsername(String username) {
this.username = username;
}
public String getUserpwd() {
return this.userpwd;
}
public void setUserpwd(String userpwd) {
this.userpwd = userpwd;
}
public Date getCreatedAt() {
return this.createdAt;
}
public void setCreatedAt(Date createdAt) {
this.createdAt = createdAt;
}
public Date getUpdatedAt() {
return this.updatedAt;
}
public void setUpdatedAt(Date updatedAt) {
this.updatedAt = updatedAt;
}
public String getRepass() {
return repass;
}
public void setRepass(String repass) {
this.repass = repass;
}
}</pre>
<p> </p>
<p><br/>修改的User-userContext-validation.xml 为</p>
<pre name='code' class='java'><?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE validators PUBLIC "-//OpenSymphony Group//XWork Validator 1.0.2//EN"
"http://www.opensymphony.com/xwork/xwork-validator-1.0.2.dtd">
<validators>
<field name="username">
<field-validator type="requiredstring">
<message>请输入用户名称</message>
</field-validator>
</field>
<field name="userpwd">
<field-validator type="requiredstring">
<message>请输入密码</message>
</field-validator>
<field-validator type="fieldexpression">
<param name="expression"><![CDATA[(userpwd==repass)]]></param>
<message>确认密码不正确</message>
</field-validator>
</field>
</validators></pre>
<p> </p>
<p><br/>User-validation.xml不用改变--------从这点我们也可以看出用Visitor校验器的好处了,就是直接在POJO里操作,相应的修改参数context属性的xml文件,很方便的</p>
<p> </p>
<p>相应的测试页面为register.jsp</p>
<p> </p>
<pre name='code' class='java'><%@ page language="java" pageEncoding="UTF-8"%>
<%@ taglib prefix="s" uri="/struts-tags"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title>用户注册页面</title>
<link href="utils/css/public.css" type="text/css" rel="stylesheet" />
</head>
<body>
<div id="all">
<div id="header">header</div>
<div id="content">
<div>已有账户,<a href="login.do">登录</a></div>
<s:form action="pro_register.do">
<s:textfield name="user.username" label="用户名称" />
<s:password name="user.userpwd" label="登录密码" />
<s:password name="user.repass" label="确认密码" />
<s:submit value="注 册" />
</s:form>
</div>
<div id="footer">footer</div>
</div>
</body>
</html></pre>
<p> </p>
<p>两次输入的密码不同,检测成功</p>
<p>确认密码不正确的截图如下:</p>
<p><br/><img src='/upload/attachment/66403/731bd882-4561-35d6-b260-6d160bf18ecf.jpg' alt=''/><br/> </p>
<p> </p>
<p> </p>
1. 我之前写的好像要是内网服务器无法访问到外网的时候 或者网络差的时候 就会抛出异常,怎么让他在脱机情况下也可以用这个validation?
2. 这个验证是在server端验证吗?
3. 如果是,为什么不考虑用js验证呢?何必让server端来承受不必要的开销呢?
1.这个问题是不应该出现的,如果真的有,我猜想可能是去网络上寻找你在xml文件开始定义的dtd文件吧?
2. 是在server端验证的,你如果在struts2源代码中关于验证表单相关的部分代码设置断点,会发现框架会把验证到的错误加到一个map里面的。
3. js验证一般只是用来验证的第一步,就像银行,银行的铁门是js,而银行装钱的保险柜就是更深层的验证。其实struts2的form标签提供了一个自动生成js代码的属性的,好像是validate="true"?只是它生成的代码不太好用,也难以阅读,所以还不如我们自己去写,这样也便于代码的理解和管理。
1. 我之前写的好像要是内网服务器无法访问到外网的时候 或者网络差的时候 就会抛出异常,怎么让他在脱机情况下也可以用这个validation?
2. 这个验证是在server端验证吗?
3. 如果是,为什么不考虑用js验证呢?何必让server端来承受不必要的开销呢?
第一个问题:我确实没遇到过这种情况,因为这个是服务器端的验证,所以我想你所说的内网访问外网或者脱机情况下也使用这个validation的话,我想是不可能的。
第二个问题:这个是Sever端的验证方式
第三个问题:JS验证当然也要用的,在实际的部署中,我们是要客户端验证和服务器端验证同时都有的。不然如果用户禁用了Javascript功能,或者通过其他的方式来Crack,我想后果是不堪设想的。至于你说的Server端不必要的开销,我想这个是正常的,不过通过客户端JS验证,基本90%的恶意或者非恶意输入产生的错误都在客户端被检测出来了,我想留个服务器的压力,在客户端和服务器端验证均存在的时候,也不是很大的。至少,在安全方面考虑,Client端和Server端的验证都要有。
<div class='quote_div'>
<p> 很多和我一样的初学者可能手边都有一般李刚写的 《Struts 2 权威指南》,</p>
</div>
<p><br/>首先特别感兴趣这句话。</p>
<p> </p>
<p>其次,该功能我在struts2下用纯annotation方式没有配置成功,struts2的doc关于annotation也是几句带过,关键就是那个context还没搞明白怎么用annotation声明,别告诉我用xml,因为我已经用了annotation,不想写xml。等高人解惑。</p>
<p> </p>
<pre name='code' class='java'> @Validations(
visitorFields = {
@VisitorFieldValidator(message = "Default message", fieldName="user.username", shortCircuit = true, appendPrefix = true),
@VisitorFieldValidator(message = "Default message", fieldName="user.password", shortCircuit = true, appendPrefix = true)
}
)</pre>
<p> </p>
1. 我之前写的好像要是内网服务器无法访问到外网的时候 或者网络差的时候 就会抛出异常,怎么让他在脱机情况下也可以用这个validation?
2. 这个验证是在server端验证吗?
3. 如果是,为什么不考虑用js验证呢?何必让server端来承受不必要的开销呢?
要学的东西太多了,而且这些框架更新又快..
当然,每个人有每个人的写法。我只是觉得在Visitor校验器比较适应于对一个POJO的复合属性进行检测,不用每次都必须重写Action里的validate方法,而且,对于在一个Action里有多个类似于execute()处理逻辑的时候,就要重写多个validateXxx()方法了。
Visitor 适用于检测Action中的复合属性,例如在一个Action里包含了User类型的属性。如果要检测密码和确认密码的功能,可以使用字段表达式校验器。或者,更为负责任的解决方法是 手动验证 ,既在Action里重写validate()方法,或者是重写validateXxx()方法。
visitor校验器,按我的理解就是:从最根本的POJO验证出发,所有的action验证都是遵循POJO里验证的规则,这样无疑会节省很多代码,而且业务逻辑更简洁明了。
这个到好,可是如果这样配了,有些action 不需要验证又怎么办呢
发表评论
-
一个异常:Caused by: java.lang.UnsupportedClassVersionError: Bad version number in .c
2013-05-14 14:17 4776今天在创建了一个工程,编译并部署到tomcat后,发现出现了 ... -
2012-03-19 11:15 关于excel 2007需要关闭两次的解决办法
2012-06-25 10:16 1134最近我发现我关EXCEL中,我是关闭最上面右角的叉,但是不能完 ... -
mysql导入数据时 USING BTREE 错误解决办法
2012-06-04 15:06 2293今天在往测试数据库导入数据时,其中一个供应商表报USING B ... -
css完美解决图片太大撑破表格 自适应图片按比例缩小
2011-10-29 00:46 1407<style type="text/css&q ... -
Mysql MyIsam引擎和InnoDB引擎的区别
2011-08-30 11:23 1126【总结】: 两种类型最主要的差别就是 InnoDB 支持事务处 ... -
Nginx 301永久重定向
2011-04-08 22:08 1173最近需要做SEO优化,将网站域名做成301永久重定向,不说了直 ... -
PHPCMS后台无法显示验证码
2011-02-24 19:55 1940查看了下php.ini文件GD库扩展已经开启了,还是显示个红X ... -
asp.net配置IIS后"服务器应用程序不可用"的解决办法 服务器应用程序不可用 您试图在此Web服务器上访问的Web应用程序当前不可用
2010-04-14 22:17 11802错误:服务器应用程序不可用 您试图在此Web服务器上访问的W ... -
Asp.net 默认配置下,Session莫名丢失的原因及解决办法
2009-05-29 10:57 1122我们平时写的asp.net程序 ... -
关于NullPointerException的引发原因
2009-05-14 19:47 3183NullPointerException,为什 ... -
addActionError与addFieldError
2009-05-13 17:03 1764addActionError() 与addFieldEr ... -
C#的显式接口和隐式接口
2009-05-01 21:03 1801接口的实现分为:隐式实现和显式实现。如果类或者结构要实现的 ... -
无法加载 DLL xpstar90.dll 的解决办法
2009-04-11 15:58 3878大家在安装VS2005后,系统自动会安装SQLEXPRESS, ... -
关于数据库外键类型的认识
2009-04-04 02:12 1236开头先贬低下自己,2年前学到数据库知识终于原原本本的还给 ... -
MySql遇到ErrorNo:28解决方法
2009-02-24 23:20 1189今天晚上接到电话,说是论坛不能访问,discuz7搭 ... -
使用cxf 关于JAXB2.1和JDK1.6/6.0版本不一致的问题 --已解决
2009-02-22 14:25 2374这段时间因为要做毕业设计,涉及到要使用WebService完 ... -
关于在MyEclipse中使用Hibernate
2009-01-02 20:39 3213在早期的 Java 数据库和 ... -
如何正确地在Axis、Axis2和Apache CXF之间抉择?
2008-12-29 21:40 1157新一代的 Web Services 框 ... -
关于rails安装mysql驱动报错问题
2008-12-20 23:46 1226我在本地用的是wamp server的一键包,只使用了m ... -
Adobe CS3 此产品的许可已经停止工作 解决方法
2008-12-16 01:19 7285【转】photoshop cs3 此产品的许可已经停止工作 解 ...
相关推荐
struts2中Visitor校验器以及域对象级别上验证和使用上下文优化的使用,我自己写的一个完整代码,详细的展示了如何使用Visitor校验器以及怎样在域对象上验证和注意事项,代码解压后可以直接部署到MyEclipse上运行,...
在探讨Struts2校验器之前,我们首先需要了解这些校验器是如何被定义和集成到框架中的。如文中所述,Struts2的内建校验器主要位于`xwork-2.0.4.jar`压缩包中的`default.xml`文件里。这个文件定义了一系列常用的数据...
以上内容详细介绍了Struts2框架的核心组成部分和技术细节,包括Action的实现方式、Result的配置方法、模型驱动的概念、异常处理机制、类型转换的实现、输入校验的流程、拦截器的设计与应用以及日志记录的配置。...
除了基本的手动编程和框架内置验证外,Struts2还支持更高级的数据验证用法,如自定义校验器和visitor校验器的应用方法等。 1. **自定义校验器**: - 可以通过扩展Struts2提供的校验器类来自定义验证规则。 - 适用...
Struts2的核心特性之一就是其强大的验证机制,该机制允许开发者在请求到达控制器之前对用户输入进行检查,确保数据的完整性和正确性,从而避免潜在的运行时错误或安全漏洞。 ### Struts2验证框架的关键属性和语法 ...
9.2 Struts中的国际化支持 185 9.3 text标签 188 9.4 i18n标签 191 9.5 以手动方式选择一个资源包 193 9.6 小结 195 第10章 Model Driven和Preparable拦截器 196 10.1 把动作与模型隔离开 196 10.2 Model ...
struts2.1.6 convertion插件(即注释方式配置)的helloworld 默认调用index()方法 文档说明用struts.xml中标签设置值,好象不用也行 rest插件例子 默认调用 create()方法 struts2.1.6 vistor校验例子 都是我测试例子,...
Struts2中的错误处理机制主要通过`addFieldError`方法实现,该方法用于在表单验证失败时添加错误信息到用户界面。例如: ```java // 添加字段级别的错误信息 addFieldError("username", "用户名不能为空"); ``` 这...
- **关于Struts2中visitor校验器的解决方法**:针对Struts2框架的验证机制,特别是visitor校验器的使用方法和常见问题的解决方案。 3. **其他技术** - **g1安装Android软件详细攻略**:为Android开发者提供了...
本文将详细介绍Struts2验证框架的配置方法以及validation.xml文件中常用的验证规则。 ### 验证框架的配置 在Struts2中,验证主要通过在Action类的关联配置文件validation.xml中定义规则来实现。该文件通常与Action...
在`Demo18_Visitor`中,可能存在`ConcreteVisitorA`和`ConcreteVisitorB`这样的类,它们分别实现`Visitor`接口中的方法,执行不同的操作。 3. **元素接口(Element)**: 元素接口定义了一个接受访问者的接收方法...
2. 定义元素接口或类,通常会有一个`accept(Visitor)`方法,接收访问者并调用相应的访问方法。 ```java public abstract class Element { public abstract void accept(Visitor visitor); } public class Employee...
9.2 Struts中的国际化支持 185 9.3 text标签 188 9.4 i18n标签 191 9.5 以手动方式选择一个资源包 193 9.6 小结 195 第10章 Model Driven和Preparable拦截器 196 10.1 把动作与模型隔离开 196 10.2 Model Driven...
Visitor模式是设计模式中的一种行为模式,它在对象结构中引入了一个访问者角色,使得访问者能够对结构中的每个元素进行操作,而不改变元素本身的行为。这种模式允许我们在不修改已有类的情况下增加新的操作,遵循了...
控制器方法可以接收请求并更新数据库中的访问记录。 3. **模型** (app/Models/): 创建一个 `Visitor` 模型,用于代表数据库中的访问记录。模型将与数据库表关联,提供插入、更新和查询数据的方法。表可能包含如 `...
"设计模式系列之visitor"是一个关于软件设计模式的讨论,特别是关注于“访问者”(Visitor)模式。这个模式是GOF(Gamma, Helm, Johnson, Vlissides)在他们的经典著作《设计模式:可复用面向对象软件的基础》中提出...
"Visitor" 在IT行业中可能指的是一个设计模式,即“访问者模式”。访问者模式是一种行为设计模式,它允许我们定义一种在多种不同对象结构中访问、改变或增加新行为的操作方式,而无需改变这些对象的结构。这个模式的...