让JSONObject支持json字符串转换为java.sql.Date(使用json-lib-2.4)

最近写案例使用到了json-lib中的JSONObject,将JavaBean转换为json字符串,以及将json字符串转换为JavaBean对象。

但是JSONObject只支持java.util.Date,而我们在JavaBean中存在java.sql.Date的属性时,会抛出异常(日志警告):

Can't transform property 'xxx' from java.lang.String into java.sql.Date. Will register a default Morpher

因此,查阅网上很多资料,有自己写一个JSONUtil类,来自己手动解析的;也有仿照源码,仿照JsonConfig的“jConfig.registerJavaPropertyNameProcessor(...)”方法,自己添加一个registerJavaValueProcessor()的方法,以在解析java.sql.Date时,自己new一个java.sql.Date返回去。

上面的方法都比较繁琐,我换了一种思路,既然JsonConfig支持设置“DateMorpher”来支持将“1997-7-1”此种字符串转换为标准的java.util.Date对象。

用于测试的Man类:

public class Man{
	private String name;
	private java.util.Date birthday;
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public java.util.Date getBirthday() {
		return birthday;
	}
	public void setBirthday(java.util.Date birthday) {
		this.birthday = birthday;
	}
}

测试类:

public class Test {
	public static void main(String[] args) {		
		String jsonStr = "{\"person\" : {\"name\" : \"张三\", \"birthday\" : \"1997-7-1\"}}";
		
		JSONObject obj = JSONObject.fromObject(jsonStr);
		
		String[] formats = {"yyyy-MM-dd"};
		
		JSONUtils.getMorpherRegistry().registerMorpher(new DateMorpher(formats));//关键之处,通过它就能够解析“1997-7-1”
		
		Man person = (Man) JSONObject.toBean(obj.getJSONObject("person"), Man.class);
		
		System.out.println("姓名:" + person.getName() + ",生日:" + person.getBirthday());
	}
}

运行后,输出结果为:姓名:张三,生日:Tue Jul 01 00:00:00 CST 1997,日期格式我们没有格式化,但证明标准的java.util.Date是能转换的!

但我们把Man中的生日修改为java.sql.Date,在运行就会报错!

public class Man{
	//......
	private java.util.Date birthday;
	//......
}

运行后报错为:

警告: Can't transform property 'birthday' from java.lang.String into java.sql.Date. Will register a default Morpher

Exception in thread "main" net.sf.json.JSONException: Error while setting property=birthday type class java.lang.String

............

然后突然想到,那么我们能不能模仿"DateMorpher"自己写一个"SqlDateMorpher"来解析为java.sql.Date呢?

最终全部代码如下:

Main.java

public class Man{
	private String name;
	private java.sql.Date birthday;
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public java.sql.Date getBirthday() {
		return birthday;
	}
	public void setBirthday(java.sql.Date birthday) {
		this.birthday = birthday;
	}
}

SqlDateMorpher.java

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.sql.Date;
import java.util.Locale;

import org.apache.commons.lang.builder.EqualsBuilder;
import org.apache.commons.lang.builder.HashCodeBuilder;

import net.sf.ezmorph.MorphException;
import net.sf.ezmorph.object.AbstractObjectMorpher;

/**
 * 本类是为了让JSONObject支持将日期字符串(如1997-6-1)转换为java.sql.Date(官方版本只支持java.util.Date)。<br>
 * 我们主要修改了原有ezmorph-1.0.6.jar包中,的“public Object morph(Object value)”方法。<br>
 * 让它在其中将解析字符串准备返回的util.Date转换为sql.Date。<br>
 * 另外,我们把本类中原有导入的“java.util.Date”修改为了“java.sql.Date”!
 * @author Administrator
 *
 */
public final class SqlDateMorpher extends AbstractObjectMorpher {
	private Date defaultValue;
	private String[] formats;
	private boolean lenient;
	private Locale locale;

	/**
	 * @param formats
	 *            a list of formats this morpher supports.
	 */
	public SqlDateMorpher(String[] formats) {
		this(formats, Locale.getDefault(), false);
	}

	/**
	 * @param formats
	 *            a list of formats this morpher supports.
	 * @param lenient
	 *            if the parsing should be lenient or not.
	 */
	public SqlDateMorpher(String[] formats, boolean lenient) {
		this(formats, Locale.getDefault(), lenient);
	}

	/**
	 * @param formats
	 *            a list of formats this morpher supports.
	 * @param defaultValue
	 *            return value if the value to be morphed is null.
	 */
	public SqlDateMorpher(String[] formats, Date defaultValue) {
		this(formats, defaultValue, Locale.getDefault(), false);
	}

	/**
	 * @param formats
	 *            a list of formats this morpher supports.
	 * @param defaultValue
	 *            return value if the value to be morphed is null.
	 * @param locale
	 *            the Locale used to parse each format.
	 * @param lenient
	 *            if the parsing should be lenient or not.
	 */
	public SqlDateMorpher(String[] formats, Date defaultValue, Locale locale, boolean lenient) {
		super(true);
		if (formats == null || formats.length == 0) {
			throw new MorphException("invalid array of formats");
		}
		// should use defensive copying ?
		this.formats = formats;

		if (locale == null) {
			this.locale = Locale.getDefault();
		} else {
			this.locale = locale;
		}

		this.lenient = lenient;
		setDefaultValue(defaultValue);
	}

	/**
	 * @param formats
	 *            a list of formats this morpher supports.
	 * @param locale
	 *            the Locale used to parse each format.
	 */
	public SqlDateMorpher(String[] formats, Locale locale) {
		this(formats, locale, false);
	}

	/**
	 * @param formats
	 *            a list of formats this morpher supports.
	 * @param locale
	 *            the Locale used to parse each format.
	 * @param lenient
	 *            if the parsing should be lenient or not.
	 */
	public SqlDateMorpher(String[] formats, Locale locale, boolean lenient) {
		if (formats == null || formats.length == 0) {
			throw new MorphException("invalid array of formats");
		}
		// should use defensive copying ?
		this.formats = formats;

		if (locale == null) {
			this.locale = Locale.getDefault();
		} else {
			this.locale = locale;
		}

		this.lenient = lenient;
	}

	public boolean equals(Object obj) {
		if (this == obj) {
			return true;
		}
		if (obj == null) {
			return false;
		}

		if (!(obj instanceof SqlDateMorpher)) {
			return false;
		}

		SqlDateMorpher other = (SqlDateMorpher) obj;
		EqualsBuilder builder = new EqualsBuilder();
		builder.append(formats, other.formats);
		builder.append(locale, other.locale);
		builder.append(lenient, other.lenient);
		if (isUseDefault() && other.isUseDefault()) {
			builder.append(getDefaultValue(), other.getDefaultValue());
			return builder.isEquals();
		} else if (!isUseDefault() && !other.isUseDefault()) {
			return builder.isEquals();
		} else {
			return false;
		}
	}

	/**
	 * Returns the default value for this Morpher.
	 */
	public Date getDefaultValue() {
		return (Date) defaultValue.clone();
	}

	public int hashCode() {
		HashCodeBuilder builder = new HashCodeBuilder();
		builder.append(formats);
		builder.append(locale);
		builder.append(lenient);
		if (isUseDefault()) {
			builder.append(getDefaultValue());
		}
		return builder.toHashCode();
	}

	//我们主要修改了此方法,以支持sql.Date
	@Override
	public Object morph(Object value) {
		if (value == null) {
			return null;
		}

		//准确匹配要转换的类型为java.sql.Date
//		if (Date.class.isAssignableFrom(value.getClass()))
		if (Date.class == value.getClass())
		{
			return (Date) value;
		}

		if (!supports(value.getClass())) {
			throw new MorphException(value.getClass() + " is not supported");
		}

		String strValue = (String) value;
		SimpleDateFormat dateParser = null;

		for (int i = 0; i < formats.length; i++) {
			if (dateParser == null) {
				dateParser = new SimpleDateFormat(formats[i], locale);
			} else {
				dateParser.applyPattern(formats[i]);
			}
			dateParser.setLenient(lenient);
			try {
				//return new dateParser.parse(strValue.toLowerCase());//原有代码返回java.util.Date
				//我们在此将它包装为java.sql.Date
				return new Date(dateParser.parse(strValue.toLowerCase()).getTime());
			} catch (ParseException pe) {
				// ignore exception, try the next format
			}
		}

		// unable to parse the date
		if (isUseDefault()) {
			return defaultValue;
		} else {
			throw new MorphException("Unable to parse the date " + value);
		}
	}

	public Class morphsTo() {
		return Date.class;
	}

	/**
	 * Sets the defaultValue to use if the value to be morphed is null.
	 *
	 * @param defaultValue
	 *            return value if the value to be morphed is null
	 */
	public void setDefaultValue(Date defaultValue) {
		this.defaultValue = (Date) defaultValue.clone();
	}

	public boolean supports(Class clazz) {
		return String.class.isAssignableFrom(clazz);
	}
} 

上面类中,全部为复制官方的DateMorpher.java,修改了构造方法的名称、java.util.Date包改为java.sql.Date,以及public Object morph(Object value)方法的局部代码(看注释),其余内容没变!

测试类Test.java

public class Test {
	public static void main(String[] args) {		
		String jsonStr = "{\"person\" : {\"name\" : \"张三\", \"birthday\" : \"1997-7-1\"}}";
		
		JSONObject obj = JSONObject.fromObject(jsonStr);
		
		String[] formats = {"yyyy-MM-dd"};
		
		JSONUtils.getMorpherRegistry().registerMorpher(new SqlDateMorpher(formats));
		
		Man person = (Man) JSONObject.toBean(obj.getJSONObject("person"), Man.class);
		
		System.out.println("姓名:" + person.getName() + ",生日:" + person.getBirthday());
	}
} 

运行结果为:“姓名:张三,生日:1997-07-01”,结果正确!

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值