一、自定义类型转换器
1.概述
Struts2提供了常规类型转换器,可以用于常用数据类型的转换,但如果目标类型是一个特殊类型,则需要自定义转换器。Struts2 类型转换器实际上都是基于OGNL实现的,在OGNL项目中,有一个TypeConverter接口,自定义类型转换器必须实现
ongl.TypeConverter。
2.编写类型转换器
(1)实现TypeConverter接口,实现一个方法
public Object convertValue(Map<String, Object> context, Object target, Member member, String propertyName, Object value, Class toType);
(2)继承DefaultTypeConverter类,重写一个方法
public Object convertValue(Map<String,Object> context,Object value,Class toType)
(3)继承StrutsTypeConverter类重写两个方法
public Object convertFromString(Map context,String[] values,Class toClass)
public String convertToString(Map context,Object o)
java.util.Date类型的属性可以接收格式为2009-07-20的请求参数值。但如果我们需要接收格式为20091221的请求参数,我们必须定义类型转换器,否则struts2无法自动完成类型转换。
第一种:继承DefalutTypeConverter类
public class DateConvert extends DefaultTypeConverter{ @Override
public Object convertValue(Map<String,Object> context,Object value,Class toType){ SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd"); try{
//String--->Date
if(toType == Date.class){
String[] params = (String[])value;
return sdf.parseObject(params[0]);
}
else if(toType == String.class){ //Date--->String
Date date = (Date)value;
return sdf.format(date);
}
}catch(ParseException e){
e.printStackTrace();
} return super.convertValue(context,value,toType);
}
}
第二种继承StrutsTypeConverter类
public class DateConvert2 extends StrutsTypeConverter{ private SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd"); //String--->Date
@Override
public Object convertFromString(Map context,String[] values,Class toClass){ if(toClass == Date.class){
try{
String date = values[0];
return sdf.parse(date);
}catch(ParseException e){
e.printStackTrace();
}
}
return null;
} //Date--->String
@Override
public String convertToString(Map context,Object o){ if(o instanceof Date){
return sdf.format(o);
} return null;
}
}
3.注册类型转换器
(1)局部注册
针对的是表单中某个字段生效。分为属性驱动方式和模型驱动方式两种。
属性驱动的方式
注意: 在Action所在的包下创建一个文件,文件的格式是:Action类名-conversion.properties文件,该文件中配置要转换数据的字段和对应的转换器全路径。birthday=com.kiwi.convert.DateConvert2
模型驱动的方式
注意: 在实体类所在的包下创建一个文件,文件的格式是:实体类名-conversion.properties文件,该文件中配置要转换数据的字段和对应的转换器全路径。
(2)全局注册
针对整个项目所有的日期类型都会生效。
在src的目录下,创建一个xwork-conversion.properties
例如: java.util.Date=cn.itcast.demo3.MyDateConverter
4.类型转换的错误处理
(1)当发生类型转换错误的时候,根据报错的信息提示,没有提供input类型的结果视图。那可以在<action>标签中配置input结果视图。
(2)如果Struts2的类型转换器执行类型转换时出现错误,该拦截器将负责将对应错误封装成表单域错误(FieldError),并将这些错误信息放入ActionContext中。
(3)使用类型转换中的错误处理用户定义Action必须继承ActionSupport。
(4)在自定义类型转换器中,异常必须抛出不能捕获,conversionError会处理该异常,然后转入名为input的逻辑视图。
(5)在Action所在包中,创建 ActionName.properties,在局部资源文件中配置提示信息
invalid.fieldvalue.属性名= 错误信息
index.jsp
<s:form action="personAction"> <s:textfield label="用户名" name="username"></s:textfield>
<s:textfield label="密 码" name="password"></s:textfield>
<s:textfield label="年 龄" name="age"></s:textfield>
<s:textfield label="生 日" name="birthday"></s:textfield> <s:submit value="提交"></s:submit>
</s:form>
struts.xml
<action name="personAction" class="com.kiwi.action.PersonAction">
<result name="success">/success.jsp</result>
<result name="input">/index.jsp</result>
</action>
结果:
二、数据校验
1.概述
数据校验分为前台校验和后台校验两种方式:
(1) JavaScript可以在前台完成校验,但是这种方式主要是提升用户的体验,通过可以绕行的方式进入到后台程序中。
(2) 后台的数据校验,在Servlet/Action中需要做数据的校验(必须要做的校验)。
Struts2提供了两种校验方式:
(1)手动编码校验。
(2)配置文件校验。
2.手动编码校验
开发步骤:
步骤一: 封装数据。
步骤二: 实现校验Action ,必须继承ActionSupport 类。
步骤三: 覆盖validate方法,完成对Action的业务方法 数据校验 this.addFieldError (ActionSupport提供)。
步骤四: 在jsp中 通过 <s:fieldError/> 显示错误信息。
(1)针对Action所有的方法进行校验
让Action继承ActionSupport类,重写ActionSupport类中的validate()方法,在该方法中完成数据校验。
//针对所有Action进行校验
@Override
public void validate(){ //用户名为空
if(StringUtils.isEmpty(person.getUsername())){ addFieldError("username","用户名不能为空");
}
//密码不能为空
if(StringUtils.isEmpty(person.getPassword())){ addFieldError("password","密码不能为空");
} }
(2)针对Action某个方法进行校验
手动在Action中编写一个方法,方法名称是validate方法名称() 例如: public void validateAdd(){ }
public String save(){
System.out.println("save()......");
return NONE;
} // 针对save方法进行校验
public void validateSave(){
// 用户名为空
if(StringUtils.isEmpty(person.getUsername())){ addFieldError("username","用户名不能为空");
}
// 密码不能为空
if(StringUtils.isEmpty(person.getPassword())){ addFieldError("password","密码不能为空");
}
}
注意:
(1)编写的是Action都需要继承ActionSupport类。
(2)代码校验不适用于大型项目,流程数据复杂时,开发量和维护量都会很大。
3.配置文件校验
(1)针对Action中所有的方法进行校验
在Action所在包中创建一个XML文件,命名规则: Action类型-validation.xml。
PersonAction-validation.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 name="username">
<field-validator type="requiredstring">
<message>用户名不能为空</message>
</field-validator>
</field> <field name="password">
<field-validator type="requiredstring">
<message>密码不能为空</message>
</field-validator> <field-validator type="stringlength">
<param name="minLength">3</param>
<param name="maxLength">12</param>
<message>密码长度必须在3~12之间</message>
</field-validator> </field> </validators>
(2)针对的是Action的某个方法进行校验
在Action所在包创建一个XML文件,命名规则: Action类名-方法名-validation.xml
4.Struts2提供的内置校验器列表
<!-- required 必填校验器 -->
<field-validator type="required">
<message>性别不能为空!</message>
</field-validator> <!-- requiredstring 必填字符串校验器 -->
<field-validator type="requiredstring">
<param name="trim">true</param>
<message>用户名不能为空!</message>
</field-validator> <!-- stringlength:字符串长度校验器 -->
<field-validator type="stringlength">
<param name="maxLength">10</param>
<param name="minLength">2</param>
<param name="trim">true</param>
<message><![CDATA[产品名称应在2-10个字符之间]]></message>
</field-validator> <!-- int:整数校验器 -->
<field-validator type="int">
<param name="min">1</param>
<param name="max">150</param>
<message>年龄必须在1-150之间</message>
</field-validator> <!-- date: 日期校验器 -->
<field-validator type="date">
<param name="min">1900-01-01</param>
<param name="max">2050-02-21</param>
<message>生日必须在${min}到${max}之间</message>
</field-validator> <!-- url: 网络路径校验器 -->
<field-validator type="url">
<message>一个有效网址</message>
</field-validator> <!-- email:邮件地址校验器 -->
<field-validator type="email">
<message>电子邮件地址无效</message>
</field-validator> <!-- regex:正则表达式校验器 -->
<field-validator type="regex">
<param name="expression"><![CDATA[^13\d{9}$]]></param>
<message>手机号格式不正确!</message>
</field-validator> <!-- fieldexpression : 字段表达式校验 -->
<field-validator type="fieldexpression">
<param name="expression"><![CDATA[(password==repassword)]]></param>
<message>两次密码输入不一致</message>
</field-validator>