admin 管理员组文章数量: 886993
Struts2学习第二天——获取参数与数据校验
文档版本 | 开发工具 | 测试平台 | 工程名字 | 日期 | 作者 | 备注 |
---|---|---|---|---|---|---|
V1.0 | 2016.06.14 | lutianfei | none |
struts2中获取请求参数
- 在struts2中action是什么?(struts2是一个mvc框架)
- View : jsp
- Model : action
- Control : action & StrutsPrepareAndExecuteFilter
1.属性驱动
1.直接将action做一个model(类似bean结构),就可以得到请求参数.
- 问题1:action封装请求参数,会不会存在线程安全问题?
- 不会:因为每一次请求,都是一个新的action。
- 缺点:需要单独定义javaBean,将action中属性copy到javaBean中。
- 优点:简单。
- 这种方式,底层是通过反射来实现的。
- 问题1:action封装请求参数,会不会存在线程安全问题?
配合第一天作业案例,修改Login1Action.java
private String username;private String password;public String getUsername() {return username;}public void setUsername(String username) {this.username = username;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}@Overridepublic String execute() throws Exception {HttpServletRequest request = ServletActionContext.getRequest();// 2.判断用户名与密码是否正确if ("tom".equals(username) && "123".equals(password)) {request.getSession().setAttribute("username", username);return SUCCESS;} else {request.setAttribute("login.message", "用户名或密码错误");return "failer";}}
}
2.在action中声明一个model。
- 在页面上使用ognl来描述
- 注意修改:
<input type="text" name="user.username">
- 优点:简单,解决了第一种封装的问题
- 缺点:在页面上使用了ognl表达式,页面不通用了。
- 问题:这种方式,数据是怎样封装的?
- 通过struts2中的interceptor进行了数据封装.
<interceptor name="params" class="com.opensymphony.xwork2.interceptor.ParametersInterceptor"/>
- 通过struts2中的interceptor进行了数据封装.
Login2Action.java
//获取请求参数 属性驱动 第二种,直接在action声明一个model
public class Login2Action extends ActionSupport {private User user;public User getUser() {return user;}public void setUser(User user) {this.user = user;}@Overridepublic String execute() throws Exception {HttpServletRequest request = ServletActionContext.getRequest();// 2.判断用户名与密码是否正确if ("tom".equals(user.getUsername()) && "123".equals(user.getPassword())) {request.getSession().setAttribute("username", user.getUsername());return SUCCESS;} else {request.setAttribute("login.message", "用户名或密码错误");return "failer";}}
}
- login2.jsp
<body>${requestScope["login.message"] }<br><form action="${pageContext.request.contextPath}/login2" method="post">username:<input type="text" name="user.username"><br>password:<input type="password" name="user.password"><br><input type="submit" value="登录"></form></body>
2.模型驱动(在开发中应用比较多)
- 开发步骤:
- 1.让action类实现 ModelDriven
- 2.重写getModel方法
- 3.在action中实例化一个model对象,让getModel方法返回这个对象。
public class Login3Action extends ActionSupport implements ModelDriven<User> { //第一步:implements ModelDriven<User>private User user = new User(); //第三步public User getModel() { //第二步
return user;
}}//jsp<body>${requestScope["login.message"] }<br><form action="${pageContext.request.contextPath}/login3" method="post">username:<input type="text" name="username"><br>password:<input type="password" name="password"><br><input type="submit" value="登录"></form></body>
- 优点:解决了属性驱动存在的问题
缺点:一次只能封装一个model对象.
struts2 有很多围绕模型驱动的特性
<interceptor name="modelDriven" class="com.opensymphony.xwork2.interceptor.ModelDrivenInterceptor"/>
为模型驱动提供了更多特性
扩展
- 1.将数据封装到List集合
页面:
username1:<input type="text" name="users[0].username"><br>
password1:<input type="password" name="users[0].password"><br>username2:<input type="text" name="users[1].username"><br>
password2:<input type="password" name="users[1].password"><br>
action类:
public class ListAction extends ActionSupport {private List<User> users;public List<User> getUsers() {return users;}public void setUsers(List<User> users) {this.users = users;}@Overridepublic String execute() throws Exception {System.out.println(users);return null;}
}//struts.xml配置
<action name="list" class="cn.itcast.action.ListAction">
</action>
- 2.将数据封装到Map集合
页面:
username1:<input type="text" name="map['aaa'].username"><br>
password1:<input type="password" name="map['aaa'].password"><br>username2:<input type="text" name="map['bbb'].username"><br>
password2:<input type="password" name="map['bbb'].password"><br>
- action类:
public class MapAction extends ActionSupport {private Map<String, User> map;public Map<String, User> getMap() {return map;}public void setMap(Map<String, User> map) {this.map = map;}@Overridepublic String execute() throws Exception {System.out.println(map);return null;}
}
struts2中提供的类型转换
- 在web中我们使用beanutils直接将表单数据封装到javaBean中。—类型转换
struts2中action得到请求参数,也可以直接封装到javaBean.
struts2 内部提供大量类型转换器,用来完成数据类型转换问题
- boolean 和 Boolean
- char 和 Character
- int 和 Integer
- long 和 Long
- float 和 Float
- double 和 Double
- Date 可以接收 yyyy-MM-dd格式字符串
- 数组 可以将多个同名参数,转换到数组中
- 集合 支持将数据保存到 List 或者 Map 集合
但并不是所有的类型都支持
- 例如:日期类型,我们传递 yyyy-MM-dd yyyy年MM月dd日格式都可以,但是如果是yyyy/MM/dd就会出现问题.
关于struts2中的类型转换器根接口是:
com.opensymphony.xwork2.conversion.TypeConverter
。
自定义类型转换器
- 操作步骤:
- 1.创建一个类实现TypeConverter接口
- 2.重写接口中方法,实现类型转换操作
- 3.注册类型转换器
1.创建一个自定义类型转换器
1.实现TypeConverter需要重写如下方法:
public Object convertValue(Map<String, Object> context, Object target, Member member, String propertyName, Object value, Class toType);
如果实现接口,这个方法参数太多(6个),不推荐。
2.继承 DefaultTypeConverter类
- 优点:重写的方法参数没有那么多
public Object convertValue(Map<String, Object> context, Object value, Class toType) {return convertValue(value, toType);}
3.推荐使用 继承
DefaultTypeConverter类
的一个子类StrutsTypeConverter
.- 原因:在这个类中将从页面传递的数据怎样封装,以及action中的数据怎样在页面上显示做了分离。
- convertFromString : 将页面数据转换至JavaBean
- convertToString : 将JavaBean转换至页面输出
public abstract Object convertFromString(Map context, String[] values, Class toClass);public abstract String convertToString(Map context, Object o);
2.注册一个自定义类型转换器
1.局部–针对于action(action中包含了一个Bean结构)
- 配置文件所在位置以及名称: 在Action类所在包 创建
Action类名-conversion.properties
- 配置文件书写格式 :
属性名称=类型转换器的全类名
- 配置文件所在位置以及名称: 在Action类所在包 创建
eg:
birthday= test.utils.MyTypeConverter
- MyTypeConverter.java
public class MyTypeConverter extends StrutsTypeConverter{@Overridepublic Object convertFromString(Map context, String[] values, Class toClass) {// System.out.println(context); //ognl.OgnlContext@faae2497// System.out.println(values[0]); //1990/11/10// System.out.println(toClass); //class java.util.DateString value = values[0];SimpleDateFormat sdf =new SimpleDateFormat("yyyy/MM/dd");Date date = null;try {date = sdf.parse(value);} catch (ParseException e) {// TODO Auto-generated catch blocke.printStackTrace();}return date;}
- RegistAction.java
public class RegistAction extends ActionSupport implements ModelDriven<User>{private User user = new User();@Overridepublic User getModel() {// return user;} @Overridepublic String execute() {System.out.println(user);return null;}
}
- sturts.xml & regist.jsp
<struts><package name="default" namespace="/" extends ="struts-default"><action name="regist" class = "test.action.RegistAction"><result name="input">/success.jsp</result></action></package>
</struts><body><form action="${pageContext.request.contextPath}/regist" method="post">username:<input type="text" name="username"><br>password:<input type="password" name="password"><br>hobby:<input type="checkbox" name="hobby" value="eat">吃<input
type="checkbox" name="hobby" value="drink">喝<input
type="checkbox" name="hobby" value="play">玩<br> age:<input
type="text" name="age"><br> birthday:<input type="text"name="birthday"><br> <input type="submit" value="注册"></form>
</body>
2.局部–针对于model(action中实例化了一个Bean)
- 配置文件所在位置以及名称: 在model类所在包 创建
model类名-conversion.properties
- 配置文件书写格式 :
属性名称=类型转换器的全类名
- 配置文件所在位置以及名称: 在model类所在包 创建
eg:在上例的基础上做如下修改
- 修改properties文件名及位置如下:
- 修改properties文件名及位置如下:
RegistAction.jsp
public class RegistAction extends ActionSupport {private String username;private String password;private int age;private Date birthday;private String[] hobby;public String[] getHobby() {return hobby;}public void setHobby(String[] hobby) {this.hobby = hobby;}public String getUsername() {return username;}public void setUsername(String username) {this.username = username;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public Date getBirthday() {return birthday;}public void setBirthday(Date birthday) {this.birthday = birthday;}@Overridepublic String execute() throws Exception {System.out.println(this.toString());return null;}@Overridepublic String toString() {return "RegistAction [username=" + username + ", password=" + password+ ", age=" + age + ", birthday=" + birthday + ", hobby="+ Arrays.toString(hobby) + "]";}}
3.全局
- 配置文件所在位置以及名称:在src下创建一个
xwork-conversion.properties
- 配置文件书写格式:
要转换的类型全名=类型转换器的全类名
- 配置文件所在位置以及名称:在src下创建一个
eg:
- 在第一个例子的基础上改成指定名字,并将其放在
src
目录下就好。 - properties文件改为:
java.util.Date=test.utils.MyTypeConverter
- 在第一个例子的基础上改成指定名字,并将其放在
注意:
对于struts2中类型转换器(非自定义),如果表单数据提交时,将数据向model封装,出现了问题,会报错:
No result defined for action cn.itcast.action.RegistAction and result input
意思是,在RegistAction的配置中没有配置input结果视图;当加入
result name=input
标签时,会跳转到对应的页面(例如success.jsp)
<action name="regist" class="cn.itcast.action.RegistAction"><result name="input">/success.jsp</result></action>
- 为什么会向input视图跳转?
- 因为struts2中的拦截器(interceptor).
- 在struts2中的
<interceptor name="conversionError" class="org.apache.struts2.interceptor.StrutsConversionErrorInterceptor"/>
用于记录类型转换问题。 - 在struts2中
<interceptor name="workflow" class="com.opensymphony.xwork2.interceptor.DefaultWorkflowInterceptor"/>
用于得到问题,向input视图跳转。
- 关于错误信息展示:
- 通过分析拦截器作用,得知当类型转换出错时,自动跳转input视图 ,在input视图页面中
<s:fieldError/>
显示错误信息
- 通过分析拦截器作用,得知当类型转换出错时,自动跳转input视图 ,在input视图页面中
如果想通过国际化改为中文显示需要在Action所在包中,创建
ActionName.properties
,在局部资源文件中配置提示信息 :invalid.fieldvalue.属性名= 错误信息
eg:
//注意加 taglib<%@ taglib prefix="s" uri="/struts-tags" %><body><s:fielderror/></body>//RegisAction.properties
invalid.fieldvalue.birthday=\u65E5\u671F\u683C\u5F0F\u8981\u6C42 yyyy/MM/dd
- 如果是自定义类型转换器,出现类型转换问题,要跳转到input视图,在类型转换器中,必须抛出异常才可以。
- eg : MyTypeConverter:第20行
public class MyTypeConverter extends StrutsTypeConverter {// 接收页面传递的数据封装到javaBean.@Overridepublic Object convertFromString(Map context, String[] values, Class toClass) {// System.out.println(context); //ognl.OgnlContext@faae2497// System.out.println(values[0]); //[Ljava.lang.String;@100c4d// System.out.println(toClass); //class java.util.DateString value = values[0];SimpleDateFormat sdf = new SimpleDateFormat("yyyy/MM/dd");Date date = null;try {date = sdf.parse(value);} catch (ParseException e) {// e.printStackTrace();throw new RuntimeException();}return date;}
关于struts2提供的数据校验
在开发中,请求参数是需要校验的。
- 客户端校验—->js
- 服务器校验—->java代码。
struts2中提供的校验—–服务器端校验。
- 分成两种:
- 1.手动校验(编码校验)
- 2.配置校验(annotation,xml) 我们讲的是xml。
1.手动校验(了解)
- 要求:action类必须继承自ActionSupport。因为需要重写一个方法
validate
- 通过测试发现在action中重写的validate方法执行了。并且是在请求处理方法(execute)之前执行的。
struts2提供的校验,也是通过拦截器实现的。
问题:在validate方法中怎样存储校验错误信息?
- 在validate方法中 this.addFieldError(Sting name,String value);
@Overridepublic void validate() {if (user.getUsername() == null|| user.getUsername().trim().length() == 0) {// 说明用户名为空this.addFieldError("username.message", "用户名不能为空");}if (user.getPassword() == null|| user.getPassword().trim().length() == 0) {this.addFieldError("password.message", "密码不能为空");}if (!(user.getAge() >= 10 && user.getAge() <= 40)) {this.addFieldError("age.message", "年龄必须在10-40之间");}// System.out.println("validate......");}
- 问题:在页面上怎样获取错误信息?(在input视图上)
<s:fielderror>
展示所有错误信息<s:fielderror fieldName="">
展示特定名称的错误信息.
<s:fielderror/><form action="${pageContext.request.contextPath}/regist" method="post">username:<input type="text" name="username"><s:fielderror fieldName="username.message"/><br>password:<input type="password" name="password"><s:fielderror fieldName="password.message"/><br>
- 问题:在同一个Action中有多个请求处理方法(login,regist)那么有些方法是需要校验的,有些是不需要的,怎样处理?
- 解决方案:创建一个名称叫
validate+请求处理方法名
- 例如:请求处理方法叫
regist()
校验的 方法名validateRegist()
- 如上情况下还有
validate
方法时,会先运行validateRegist()
方法,后运行validate()
方法。
- 解决方案:创建一个名称叫
2.配置校验(xml,常用)
- struts2的校验框架。
已经完成了校验操作(做了很多校验方法)。而我们在使用时,只需要将它们调用就可以(通过配置文件)
要求:action类必须继承自ActionSupport类。
校验配置文件的配置方法
- 位置:xml文件要与action类在同一个包下
- 名称:
action类名-validation.xml
- 约束:
xwork-core-2.3.7.jar
中xwork-validator-1.0.3.dtd
下
<!DOCTYPE validators PUBLIC
"-//Apache Struts//XWork Validator 1.0.3//EN"
".0.3.dtd">
- 书写:
- 1.根元素
<validators>
- 2.子元素
<field name="属性名称"></field>
- 3.
<field>子元素
- 指定校验器
<field-validator type="校验器">
- 问题:校验器有哪些?
- 校验器位置:
xwork-core-2.3.7.jar
中 /com/opensymphony/xwork2/validator/validators/default.xml下
- 指定校验器
- 1.根元素
<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"/>
- 4.
<field-validator>子元素
<message>错误信息</message>
5.
<field-validator>
子元素<param name="">值</param>
: 用于指定校验器中的参数。
eg :
<validators><!-- 对username属性进行校验 --><field name="username"><!-- 指定username不能为空 --><field-validator type="requiredstring"><!-- 错误信息 --><message>用户名不能为空--------</message></field-validator>
</validators>
配置校验中的常用校验器
required
(必填校验器,要求被校验的属性值不能为null)requiredstring
(必填字符串校验器,要求被校验的属性值不能为null,并且长度大于0,默认情况下会对字符串去前后空格),如果不想去空格,只需如下设置。
stringlength
(字符串长度校验器,要求被校验的属性值必须在指定的范围内,否则校验失败- minLength参数指定最小长度
- maxLength参数指定最大长度
- trim参数指定校验field之前是否去除字符串前后的空格
regex
正则表达式校验器,检查被校验的属性值是否匹配一个正则表达式,expression参数指定正则表达式,caseSensitive参数指定进行正则表达式匹配时,是否区分大小写,默认值为true)
int(整数校验器,要求field的整数值必须在指定范围内
- min指定最小值
- max指定最大值
double
(双精度浮点数校验器,要求field的双精度浮点数必须在指定范围内,min指定最小值,max指定最大值)fieldexpression
(字段OGNL表达式校验器,要求field满足一个ognl表达式,expression参数指定ognl表达式,该逻辑表达式基于ValueStack进行求值,返回true时校验通过,否则不通过)
email
(邮件地址校验器,要求如果被校验的属性值非空,则必须是合法的邮件地址)
url(网址校验器,要求如果被校验的属性值非空,则必须是合法的url地址)
date
:日期校验器,要求field的日期值必须在指定范围内- min指定最小值
- max指定最大值
问题:通过配置校验,怎样处理在同一个action中存在多个请求处理方法校验问题?
- 只需要将校验xml文件名称修改就可以。
- 原名称:
action类名-valication.xml
现在要对action类中某一个方法校验。 - 修改名称:
action类名-action名称-validation.xml
- 原名称:
- 只需要将校验xml文件名称修改就可以。
本文标签: Struts2学习第二天获取参数与数据校验
版权声明:本文标题:Struts2学习第二天——获取参数与数据校验 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.freenas.com.cn/jishu/1732357645h1534897.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论