今天跑程序的时候莫名其妙的出现了下面的一个异常:

java.lang.NoSuchMethodException:com.ca.agent.model.mybatis.ApprovalInforCangra.setSubDate([Ljava.lang.String;)

这类异常信息在以前是处理过的,按照以前的思路在eclipse的调试模式下看看具体的情况,不过这次比较奇怪,根本没进入到对应的Action类中就抛出了此异常信息,进一步查找发现,在调用拦截器方法的时候就抛出了此异常信息,还有没有想明白的就是对应的功能我是测试过好多次的,之前这一块是没有任何问题的!不过现在事实是却抛出了异常信息,一定是那里出错了,到底是那里哪?查找了大半个下午也没找到对应的解决方案,应该是我忽略了什么地方,一时半会可能也想不起来了,找位同事帮忙吧!于是找来同事G来帮忙,最终找到了问题的原因!Struts2默认可支持String和Date的转换,但仅支持短格式和本地化有关日期格式转换,前几天我将自己的IE浏览器的语言环境改成了英语的!而对于yyyy-MM-dd这种日期类型,在英语语言中是没法匹配的,由于Struts2匹配日期时,使用了Locale,所以使用英文的语言环境,就导致了此问题的出现,把中文设置为默认语言环境后,再测试,没问题了!

Web开发会涉及到很多类型转换的情况。我们知道,页面中的一切值都是字符串类型,而到后台,我们需要的可能是其他各种类型;同时,页面显示也是字符串类型。这就涉及到Web中基本的类型转换问题:从String转换为各种类型与从各种类型转换为String类型。

在Java Web开发中,进行上述转换一般有以下几种:

1、在Servlet中,这一切的转换我们得自己写代码完成;

2、在Struts1.x中,我们通过apache commons-beanutils中的converters来帮助完成这些事情;

3、在Struts2中,使用的则是基于ongl的类型转换;

……

由 于类型转换的通用性,因而Web框架都会实现大多数类型的转换功能,而不需要程序员编码实现。然而,对于java.util.Date这种类型的转换,各 大框架似乎做得都不尽如人意。如:在Struts1.x中,该类型的转换就会有问题,很多人建议使用java.sql.Date这种类型来解决日期转换的 问题(实际上可以自定义一个类型转换器来解决该问题)。在Struts2中,这个问题似乎依然存在,也许你从来没有遇到过。的确,一般人确实不会遇到,会 觉得没有这个问题。下面就是我遇到的问题与解决方法。

日期类型的转换,Web开发中几乎都会遇到,我现在做得项目也不例外。在开发的过程 中,也许就像你一样,我没有对日期类型的转换做任何特殊的处理,而且Struts2也很好的帮我完成了转换。然而同事测试的时候却出现了一个“莫名其妙” 的问题:输入一个常用格式的日期类型yyyy-MM-dd,到后台却报错:找不到对应的set方法—— setEffDate(Ljava.lang.String)。的确,程序中只有setEffDate(java.util.Date)这个方法,没有 setEffDate(Ljava.lang.String)这个方法。从Ljava.lang.String可以看出,传到后台的String类型并没 有转换成Date类型,因而报错。

一开始,我以为是我UT没做好,于是在自己的电脑上模拟同事的测试,结果一点问题也没有。这就奇怪了。 经过自己分析,觉得可能是IE浏览器的原因,因为同事测试用的是IE,而我用的是FireFox。于是在自己的机子上用IE测试,同时在同事机子上用 FireFox测试,结果这两次测试都没有出现上面的问题。虽然没有找到问题所在,但可以初步肯定:IE的问题,但似乎又不完全是IE的问题,因为在我的 电脑上的IE(版本与同事一样,都是IE6)没有上述问题。这就奇怪了,是什么问题呢,真是百思不得其解。

这个时候,我想起了之前遇到的 一个不解得情况:从后台获得的日期类型在页面上显示时,跟上面情况一样,在同事的IE中,日期显示的格式竟然是:yyyy-MM- ddTHH:mm:ss。多了一个T,真是莫名其妙,而且只在同事的IE浏览器中出现(当时解决方法是在JS中将'T'替换为空格,没有去深究,但现在必 须的深究了)。yyyy-MM-ddTHH:mm:ss这种日期格式有吗?于是查询JDK:在SimpleDateFormat类中找到了该日期格式,这 种格式是“美国语言环境中日期和时间的模式之一”。原来还真有这种格式。竟然这是美国语言中使用的日期格式,而Struts2是美国人开发的,也许跟这个 有点关系。于是查看Struts2中关于Date类型转换的源码。

在XWorkBasicConverter类中

private Object doConvertToDate(Map context, Object value, Class toType) {

Date result = null;

if (value instanceof String && value != null && ((String) value).length() > 0) {

String sa = (String) value;

Locale locale = getLocale(context);

DateFormat df = null;

if (java.sql.Time.class == toType) {

df = DateFormat.getTimeInstance(DateFormat.MEDIUM, locale);

} else if (java.sql.Timestamp.class == toType) {

Date check = null;

SimpleDateFormat dtfmt = (SimpleDateFormat) DateFormat.getDateTimeInstance(DateFormat.SHORT,

DateFormat.MEDIUM,

locale);

SimpleDateFormat fullfmt = new SimpleDateFormat(dtfmt.toPattern() + MILLISECOND_FORMAT,

locale);

SimpleDateFormat dfmt = (SimpleDateFormat) DateFormat.getDateInstance(DateFormat.SHORT,

locale);

SimpleDateFormat[] fmts = {fullfmt, dtfmt, dfmt};

for (SimpleDateFormat fmt : fmts) {

try {

check = fmt.parse(sa);

df = fmt;

if (check != null) {

break;

}

} catch (ParseException ignore) {

}

}

} else if (java.util.Date.class == toType) {

Date check = null;

DateFormat[] dfs = getDateFormats(locale);

for (DateFormat df1 : dfs) {

try {

check = df1.parse(sa);

df = df1;

if (check != null) {

break;

}

}

catch (ParseException ignore) {

}

}

}

//final fallback for dates without time

if (df == null) {

df = DateFormat.getDateInstance(DateFormat.SHORT, locale);

}

try {

df.setLenient(false); // let's use strict parsing (XW-341)

result = df.parse(sa);

if (!(Date.class == toType)) {

try {

Constructor constructor = toType.getConstructor(new Class[]{long.class});

return constructor.newInstance(new Object[]{Long.valueOf(result.getTime())});

} catch (Exception e) {

throw new XWorkException("Couldn't create class " + toType + " using default (long) constructor", e);

}

}

} catch (ParseException e) {

throw new XWorkException("Could not parse date", e);

}

} else if (Date.class.isAssignableFrom(value.getClass())) {

result = (Date) value;

}

return result;

}

private DateFormat[] getDateFormats(Locale locale) {

DateFormat dt1 = DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.LONG, locale);

DateFormat dt2 = DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.MEDIUM, locale);

DateFormat dt3 = DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.SHORT, locale);

DateFormat d1 = DateFormat.getDateInstance(DateFormat.SHORT, locale);

DateFormat d2 = DateFormat.getDateInstance(DateFormat.MEDIUM, locale);

DateFormat d3 = DateFormat.getDateInstance(DateFormat.LONG, locale);

DateFormat rfc3399 = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss");

DateFormat[] dfs = {dt1, dt2, dt3, rfc3399, d1, d2, d3}; //added RFC 3339 date format (XW-473)

return dfs;

}

其中SHORT、MEDIUM、LONG在JDK中的DateFormat类中有说明。

从 这句DateFormat rfc3399 = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss");可以看出,Struts2硬编码使用了这样一种格式。然 而,Struts2中这种格式是放在最后的,为啥只有同事的IE浏览器测试时使用的是这种格式呢?(在调试时,用同时IE,日期输入中按这种格式输入,还 真的没有问题了)这说明,同事的电脑中,前面六种DateFormat都没有匹配,查看DateFormat中关于SHORT、MEDIUM、LONG的 说明,可以知道,对于yyyy-MM-dd这种日期类型,在英语语言中是没法匹配的,由于Struts2匹配日期时,使用了Locale,可见,同事的IE浏览器默认的语言环境是英语。一经查看,果然如此,把中文设置为默认语言环境,再测试,没问题了。终于知道了原因。

个人觉得,Struts2中,最后一种日期模式写死成美国标准,不是很好。

针 对这个问题,我们没法要求客户一定设置中文为默认浏览器的语言环境。因而对于Date类型的转换,可以自己定义一个转换器。以下来自 http://www.javaeye.com/wiki/struts2/1365-passing-parameters-in-struts2 中的一个类型转换器定义(不适合国际化的环境),如需要,你可以定义自己的转换器:

public class DateConverter extends DefaultTypeConverter {

private static final Logger logger = Logger.getLogger(DateConverter.class);

private static final String DATETIME_PATTERN = "yyyy-MM-dd HH:mm:ss";

private static final String DATE_PATTERN = "yyyy-MM-dd";

private static final String MONTH_PATTERN = "yyyy-MM";

/**

* Convert value between types

*/

@SuppressWarnings("unchecked")

public Object convertValue(Map ognlContext, Object value, Class toType) {

Object result = null;

if (toType == Date.class) {

result = doConvertToDate(value);

} else if (toType == String.class) {

result = doConvertToString(value);

}

return result;

}

/**

* Convert String to Date

*

* @param value

* @return

*/

private Date doConvertToDate(Object value) {

Date result = null;

if (value instanceof String) {

result = DateUtils.parseDate((String) value, new String[] { DATE_PATTERN, DATETIME_PATTERN, MONTH_PATTERN });

// all patterns failed, try a milliseconds constructor

if (result == null && StringUtils.isNotEmpty((String)value)) {

try {

result = new Date(new Long((String) value).longValue());

} catch (Exception e) {

logger.error("Converting from milliseconds to Date fails!");

e.printStackTrace();

}

}

} else if (value instanceof Object[]) {

// let's try to convert the first element only

Object[] array = (Object[]) value;

if ((array != null) && (array.length >= 1)) {

value = array[0];

result = doConvertToDate(value);

}

} else if (Date.class.isAssignableFrom(value.getClass())) {

result = (Date) value;

}

return result;

}

/**

* Convert Date to String

*

* @param value

* @return

*/

private String doConvertToString(Object value) {

SimpleDateFormat simpleDateFormat = new SimpleDateFormat(DATETIME_PATTERN);

String result = null;

if (value instanceof Date) {

result = simpleDateFormat.format(value);

}

return result;

}

}

可以将该转换器注册为全局的:在classpath下建立xwork-conversion.properties文件,内容为:java.util.Date=你的类型转换器的完整限定类名

参看:

java中struts2日期_Struts2中Date日期转换的问题相关推荐

  1. Java常用API(六)Date 日期类介绍及使用

    一.简单介绍 所在包 : java.util.Date. Date d = new Date(); System.out.println(d); // 打印出的内容是: 月份 日期 时间 时区  年份 ...

  2. Java中怎样遍历两个Date日期之间的每一天

    场景 前端传递了两个时间参数,一个开始时间一个结束时间. 后端获取到了这两个时间后怎样遍历这个时间段,并获取这个时间段中的每一天. 注: 博客: https://blog.csdn.net/badao ...

  3. java form的时间格式_java date日期格式

    packagedateformat;import java.util.*;/*** Title: 日期转换函数 * Description: 有以下功能方法可以使用 * function 1:date ...

  4. java web初始化实例_Struts2中实现web应用的初始化实例详解

    Struts2中实现web应用的初始化实例详解 在JavsSE中,main方法为应用提供了入口,而在Android中,我们可以使用Application对于整个应用的生命周期进行管理,那么在基于Str ...

  5. 【Java学习笔记】字符串和Date的转换

    Date date = new Date();System.out.println("当前时间对象:" + date);//Date对象转为本地时间格式SimpleDateForm ...

  6. 面试题1 -- Java 中,怎么在格式化的日期中显示时区?

    使用SimpleDateFormat来实现格式化日期 import java.text.SimpleDateFormat; import java.util.Date;public class Dat ...

  7. java date 日期部分_Java处理Date时间格式的各种场景工具类(二)

    public class DateUtil { public enum DateType { YEAR, MONTH, DAY, HH, MI, SS, YYYY_MM_DD, YYYYMMDD } ...

  8. oracle 转number日期,oracle number型日期转date型日期

    在搞数据库时,发现有这样的一个字段,类型是NUMBER(38),查看了一下里面的数据,都是这样的, 1323957678114 1321326994295 1322129306804 13221297 ...

  9. java日期工具类、日期格式校验、日期格式化

    文章目录 前言 日期格式校验 String转Date Date格式化为String类型 获取指定日期所在季度的第一天 获取指定日期所在季度的最后一天 LocalDate转Date 日期utils工具类 ...

最新文章

  1. ip别名删除第一个,其余别名就自动删除的分析
  2. asp.net mvc连接mysql_asp.net mvc4连接mysql
  3. 花了几百万,创业一年学到了什么
  4. Qt版本中国象棋开发(二)
  5. 洛谷UVA1328,POJ1961-Period【KMP,字符串】
  6. netty SimpleChannelInboundHandler类继承使用
  7. SAP License:物料编码原则<外部分配还是内部分配>之三
  8. Flask源码解析:从第一个版本开始阅读Flask源码
  9. git diff生成patch用法
  10. java 键盘钩子程序
  11. 第四届组织医学与再生医学国际学术研讨会(CTERM-BT 2019)
  12. fake_useragent导入无法引用的解决方法
  13. SpringBoot的Upd服务端,客户端案列
  14. 以“万物零售”为主题第十届中国网上零售年会
  15. 陶哲轩实分析 5.2 节习题试解
  16. 《一舞醉红尘,一笑歌岁月》
  17. 解决谷歌浏览器chrome的CPU占用率过高的问题
  18. 获取百度首页的源代码
  19. 实验吧-因缺思汀的绕过WriteUp
  20. 删除控制文件中的磁带备份信息

热门文章

  1. Spring MVC的优势
  2. MySQL查询指定字段
  3. 【LeetCode】0938. 二叉搜索树的范围和(二叉树的遍历)
  4. 【SpringMVC 之应用篇】 2_SpringMVC 的传参问题
  5. 这 56 个代码注释让我笑吐了
  6. java网络编程(三)
  7. svn教程----TortoiseSVN常用操作
  8. struts2教程(10)
  9. struts2的OGNL表达式(三)
  10. JQuery获取元素的N种方法