-----------------------------------------------------------------------------------------
从jsp页面传到action的值都为String类型的,struts有默认的类型转换器,会自动转化为一些基本类型如int,double等等,而默认的Date类型转换需要严格输入浏览器的地区国家时间才正确转换,所以需要自定义通用的时间转换格式,而且系统的java.util.Date格式有很大的缺陷,不能检查非法的日期,所以要实现自定义的时间类型Date的转换器:
过程如下:
1.先创建一个类,该类要继承基类StrutsTypeConverter,然后需要复写副父类的两个方法:
public Object convertFromString(Map context, String[] values, Class toClass);
public String convertToString(Map context, Object o);
从字面意思就知道两个方法的作用,第一个是从jsp传来的String类型转换为自定义的Class类型(参数名称就是toClass),第二个是从action中返回到jsp页面时也要把jsp解析不了的类转化为String类型
注:
1.老师说在进行转换时第一个需要检查Class是否为Date类型,但我感觉不需要,因为在后面的配置中已经明确说明了是转换为Date类型,在这再次检查感觉多余(去掉检查后测试没问题,执行的是我的转换器的代码)
2.老师说values在操作时需要检查是否为空?我测试时是不用检查的,我定义了两个Date域,只有在action的set方法名和jsp的域名同时存在时,才会调用该转换器的方法,否则是不会调用的,我在里面输出任何东西都没显示,证明没进入到该函数中
3.注意values有个length属性,这个属性并不是需要转换的类型的个数,这个属性的值是与jsp的域名想对应的,我测试过,jsp中有几个不同的域名,该转换器的函数就调用几次(即使不输入内容),而相同的域出现n次时,调用这个域的转换器的函数中的values.length的值就为n,所以valuse.length的值不能用来判断是否输入,我感觉values.length的值肯定大于1,因为如果小于1,就不会进入该转换器
4.老师在数要判断是否输入时,用的是if(input == null || input.equals("")),其实检查的方法是values[i].equals(""),这个稍微写sysout就能出结果了,其实我觉得如果是input==null,那么values.length的是否为0呢????
5.一个域一般在jsp页面上只出现一次,但如果出现多次,那么如果所有的date都需要转换时,需要检查所有的date类型是否都正确,如果都正确,才返回date的值,但有一个问题,那就是返回哪一个值呢?其实返回哪一个值只能由程序员自己决定,因为jsp页面上的相同域只能对应action的一个属性,那么设计多个域没什么作用,如果是再次确认日期,那么可以用校验器验证就可以了。
2.写完类型转换器的类,下一步就是要配置属性文件,这个好理解,你实现了转换器,没告诉编译器,没告诉struts框架,谁来帮你转换呢?
创建一个xwork-conversion.properties文件(文件名是固定的),在里面添加一行代码,把你要转换的代码与你写的代码划等号就行了:
java.util.Data=com.struts2.converter.DateTypeConverter
注:
1.文件一般放在src目录下
2.写法是包名+类名
写完貌似要重启tomcat才能生效
我的DateTypeConverter代码如下(重新检测了日期额度合法性,因为format()方法不会直接检测出不合法的日期,如2012-50-2也能过),有什么正确的请指出,谢谢:
从jsp页面传到action的值都为String类型的,struts有默认的类型转换器,会自动转化为一些基本类型如int,double等等,而默认的Date类型转换需要严格输入浏览器的地区国家时间才正确转换,所以需要自定义通用的时间转换格式,而且系统的java.util.Date格式有很大的缺陷,不能检查非法的日期,所以要实现自定义的时间类型Date的转换器:
过程如下:
1.先创建一个类,该类要继承基类StrutsTypeConverter,然后需要复写副父类的两个方法:
public Object convertFromString(Map context, String[] values, Class toClass);
public String convertToString(Map context, Object o);
从字面意思就知道两个方法的作用,第一个是从jsp传来的String类型转换为自定义的Class类型(参数名称就是toClass),第二个是从action中返回到jsp页面时也要把jsp解析不了的类转化为String类型
注:
1.老师说在进行转换时第一个需要检查Class是否为Date类型,但我感觉不需要,因为在后面的配置中已经明确说明了是转换为Date类型,在这再次检查感觉多余(去掉检查后测试没问题,执行的是我的转换器的代码)
2.老师说values在操作时需要检查是否为空?我测试时是不用检查的,我定义了两个Date域,只有在action的set方法名和jsp的域名同时存在时,才会调用该转换器的方法,否则是不会调用的,我在里面输出任何东西都没显示,证明没进入到该函数中
3.注意values有个length属性,这个属性并不是需要转换的类型的个数,这个属性的值是与jsp的域名想对应的,我测试过,jsp中有几个不同的域名,该转换器的函数就调用几次(即使不输入内容),而相同的域出现n次时,调用这个域的转换器的函数中的values.length的值就为n,所以valuse.length的值不能用来判断是否输入,我感觉values.length的值肯定大于1,因为如果小于1,就不会进入该转换器
4.老师在数要判断是否输入时,用的是if(input == null || input.equals("")),其实检查的方法是values[i].equals(""),这个稍微写sysout就能出结果了,其实我觉得如果是input==null,那么values.length的是否为0呢????
5.一个域一般在jsp页面上只出现一次,但如果出现多次,那么如果所有的date都需要转换时,需要检查所有的date类型是否都正确,如果都正确,才返回date的值,但有一个问题,那就是返回哪一个值呢?其实返回哪一个值只能由程序员自己决定,因为jsp页面上的相同域只能对应action的一个属性,那么设计多个域没什么作用,如果是再次确认日期,那么可以用校验器验证就可以了。
2.写完类型转换器的类,下一步就是要配置属性文件,这个好理解,你实现了转换器,没告诉编译器,没告诉struts框架,谁来帮你转换呢?
创建一个xwork-conversion.properties文件(文件名是固定的),在里面添加一行代码,把你要转换的代码与你写的代码划等号就行了:
java.util.Data=com.struts2.converter.DateTypeConverter
注:
1.文件一般放在src目录下
2.写法是包名+类名
写完貌似要重启tomcat才能生效
我的DateTypeConverter代码如下(重新检测了日期额度合法性,因为format()方法不会直接检测出不合法的日期,如2012-50-2也能过),有什么正确的请指出,谢谢:
package com.struts2.converter;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Map;
import org.apache.struts2.util.StrutsTypeConverter;
import com.opensymphony.xwork2.conversion.TypeConversionException;
public class DateTypeConverter extends StrutsTypeConverter{
@Override
public Object convertFromString(Map context, String[] values, Class toClass) {
Date date = null;
if (toClass == Date.class) {
for (int i = 0; i < values.length; i ++) {
String input = values[i];
if (!input.equals("")) {
String[] deli = new String[]{"/","-"};
int index;
for (index = 0; index < deli.length; index ++) {
if (input.indexOf(deli[index]) > 0)
break;
}
if (index < deli.length) {
String temp = input;
int j = temp.indexOf(deli[index]);
String yearString = temp.substring(0, j);
temp = temp.substring(j+1);
j = temp.indexOf(deli[index]);
String monthString = temp.substring(0, j);
temp = temp.substring(j+1);
String dayString = temp;
int year, month, day;
try {
year = Integer.parseInt(yearString);
month = Integer.parseInt(monthString);
day = Integer.parseInt(dayString);
} catch (TypeConversionException e) {
throw new TypeConversionException();
}
if (checkDate(year, month, day) == true) {
SimpleDateFormat sdf = new SimpleDateFormat("yyyy" + deli[index] + "MM" + deli[index] + "dd");
try {
date = sdf.parse(input);
continue;
} catch (ParseException e) {
throw new TypeConversionException();
}
}
}
}
throw new TypeConversionException();
}
}
return date;
}
private boolean checkDate(int year, int month, int day) {
if (year < 0 || month < 0 || month > 12) {
throw new TypeConversionException();
}
int[][] arr = {{31,28,31,30,31,30,31,31,30,31,30,31},
{31,29,31,30,31,30,31,31,30,31,30,31}};
for (int index = 0; index < 12; index ++) {
if (day > arr[leapYear(year)][index] || day < 0) {
throw new TypeConversionException();
}
}
return true;
}
private int leapYear(int year) {
if (year % 100 != 0 && year % 4 == 0) {
return 1;
} else if (year % 400 == 0) {
return 1;
}
return 0;
}
@Override
public String convertToString(Map context, Object o) {
if (o == null) {
return null;
}
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
return sdf.format((Date)o);
}
}