关于JSONObject的封装,或者说使用,现在市面上很多。这里不做过多的描述,但是有种情况却不得不说明下,JSONObject进行对对象进行JSON格式转换,但是在转换过程中,遇到了
Java.sql.Date类型的属性无法完成转换,并且抛出异常:net.sf.json.JSONException:
java.lang.reflect.InvocationTargetException
很多人遇到这个问题后,应该会查询百度等搜索引擎,那么可能得到一种类型转换的说法,我们也得到这样的说法,
后来多方测试,也确实是这个问题。如何解决?
或许很多人会说,那既然时间格式无法转换,我们可以转换设计类型嘛,数据库中我们不用date或datetime,直接用
varchar,而java中直接用String好了。确实这不失一个解决问题的办法,但是如果我们不改呢?
下面是我给出的设计图:
在这个设计图中,我给出了一个接口JsonValueProcessor ,这个接口可以自定义一些JSON类型转换器,正好,我就
分别定义了3种不同类型的类型转换器。
分析上图,我定义了3种角色:
1、类型转换器抽象接口:分别定义了2个接口方法,一个用于处理数组,一个用于处理属性类型;
2、类型转换器具体实现类:实现了上述抽象接口类的接口方法;
3、调用者:用户通过调用“调用者”的方法,完成由对象向JSONObject转换。
类型转换器抽象接口,由json-lib.jar提供,我们不必定义。
处理java.sql.Date类型属性的类型转换器:
-
package com.lovo.util;
-
-
import java.text.SimpleDateFormat;
-
import java.util.Date;
-
-
import net.sf.json.JsonConfig;
-
import net.sf.json.processors.JsonValueProcessor;
-
-
-
-
-
-
-
public class SQLDateProcessor implements JsonValueProcessor{
-
-
private String format = "yyyy-MM-dd hh:mm:ss";
-
public SQLDateProcessor() {
-
super();
-
-
}
-
-
public SQLDateProcessor(String format) {
-
this.format = format;
-
}
-
-
public Object processArrayValue(Object arg0, JsonConfig arg1) {
-
-
return arg0;
-
}
-
-
-
-
-
public Object processObjectValue(String str, Object obj, JsonConfig arg2) {
-
-
String ret = "";
-
if(obj instanceof java.sql.Date){
-
SimpleDateFormat sdf = new SimpleDateFormat(format);
-
ret = sdf.format(new Date(((java.sql.Date) obj).getTime()));
-
}
-
return ret;
-
}
-
-
}
处理java.util.Date类型的类型转换器:
-
package com.lovo.util;
-
-
import java.text.SimpleDateFormat;
-
import java.util.Date;
-
-
import net.sf.json.JsonConfig;
-
import net.sf.json.processors.JsonValueProcessor;
-
-
-
-
-
-
-
public class UtilDateProcessor implements JsonValueProcessor{
-
-
private String format = "yyyy-MM-dd hh:mm:ss";
-
public UtilDateProcessor() {
-
super();
-
-
}
-
-
public UtilDateProcessor(String format) {
-
this.format = format;
-
}
-
-
public Object processArrayValue(Object arg0, JsonConfig arg1) {
-
-
return arg0;
-
}
-
-
-
-
-
public Object processObjectValue(String str, Object obj, JsonConfig arg2) {
-
-
String ret = "";
-
if(obj instanceof java.util.Date){
-
SimpleDateFormat sdf = new SimpleDateFormat(format);
-
ret = sdf.format(((Date) obj).getTime());
-
}
-
return ret;
-
}
-
-
}
处理java.sql.Timestamp类型的类型转换器:
-
package com.lovo.util;
-
-
import java.text.SimpleDateFormat;
-
import java.util.Date;
-
-
import net.sf.json.JsonConfig;
-
import net.sf.json.processors.JsonValueProcessor;
-
-
-
-
-
-
-
public class TimestampProcessor implements JsonValueProcessor{
-
-
private String format = "yyyy-MM-dd hh:mm:ss";
-
public TimestampProcessor() {
-
super();
-
-
}
-
-
public TimestampProcessor(String format) {
-
this.format = format;
-
}
-
-
public Object processArrayValue(Object arg0, JsonConfig arg1) {
-
-
return arg0;
-
}
-
-
-
-
-
-
public Object processObjectValue(String str, Object obj, JsonConfig arg2) {
-
-
String ret = "";
-
if(obj instanceof java.sql.Timestamp){
-
SimpleDateFormat sdf = new SimpleDateFormat(format);
-
ret = sdf.format(((Date) obj).getTime());
-
}
-
return ret;
-
}
-
-
}
调用者类:
-
package com.lovo.util;
-
-
import java.util.Iterator;
-
import java.util.Map;
-
-
import net.sf.json.JSONObject;
-
import net.sf.json.JsonConfig;
-
import net.sf.json.processors.JsonValueProcessor;
-
-
-
-
-
-
public class JSONUtil {
-
-
-
-
-
-
-
-
public static JSONObject toJson(Object obj) {
-
return JSONObject.fromObject(obj);
-
}
-
-
-
-
-
-
-
-
-
public static JSONObject toJson(Object obj,Map<String,JsonValueProcessor> processors) throws ClassNotFoundException{
-
-
JsonConfig config = new JsonConfig();
-
if(processors != null && !processors.isEmpty()){
-
Iterator<java.util.Map.Entry<String, JsonValueProcessor>> it = processors.entrySet().iterator();
-
while (it.hasNext()) {
-
Map.Entry<java.lang.String, net.sf.json.processors.JsonValueProcessor> entry = (Map.Entry<java.lang.String, net.sf.json.processors.JsonValueProcessor>) it
-
.next();
-
String key = entry.getKey();
-
JsonValueProcessor processor = processors.get(key);
-
-
Class<?> cls = Class.forName(key);
-
config.registerJsonValueProcessor(cls, processor);
-
}
-
}
-
return JSONObject.fromObject(obj, config);
-
}
-
}
客户端调用“调用者”类,来完成对象向JSONObject进行转换:
-
package com.test.util;
-
-
import java.sql.Date;
-
import java.sql.Timestamp;
-
import java.util.HashMap;
-
import java.util.Map;
-
-
import org.junit.Ignore;
-
import org.junit.Test;
-
-
import com.lovo.util.SQLDateProcessor;
-
import com.lovo.util.JSONUtil;
-
import com.lovo.util.TimestampProcessor;
-
import com.lovo.util.User;
-
-
import net.sf.json.JSONObject;
-
import net.sf.json.processors.JsonValueProcessor;
-
-
public class JSONTest {
-
@Test
-
public void testJsonObjectOne() {
-
String shortFormat = "yyyy-MM-dd";
-
String longFormat = "yyyy-MM-dd hh:mm:ss";
-
-
Date sqlDate = new Date(System.currentTimeMillis());
-
Timestamp createTime = new Timestamp(System.currentTimeMillis());
-
User user = new User("高高", sqlDate, createTime);
-
-
-
Map<String, JsonValueProcessor> processors = new HashMap<String, JsonValueProcessor>();
-
-
-
processors.put("java.sql.Date", new SQLDateProcessor(shortFormat));
-
-
-
-
processors.put("java.sql.Timestamp", new TimestampProcessor(longFormat));
-
-
JSONObject json = null;
-
try {
-
json = JSONUtil.toJson(user, processors);
-
} catch (ClassNotFoundException e) {
-
-
e.printStackTrace();
-
}
-
-
System.out.println(json.toString());
-
}
-
-
-
-
-
@Ignore
-
public void testJsonObjectTwo() {
-
-
String str = "{'createTime':'2016-06-03 04:05:23','birthday':'2016-06-03','name':'1'}";
-
JSONObject json = null;
-
try {
-
json = JSONUtil.toJson(str, null);
-
} catch (ClassNotFoundException e) {
-
-
e.printStackTrace();
-
}
-
-
System.out.println(json.get("name"));
-
System.out.println(json.get("createTime"));
-
System.out.println(json.get("birthday"));
-
}
-
-
}
得到的结果是:
-
{"createTime":"2016-06-03 05:09:49","birthday":"2016-06-03","name":"高高"}
在JSONUtil类中,由于我们可以采用JSONConfig类来一次性注册多个类型转换器,所以我将多个类型转换器装配到
Map中,迭代Map集合采用反射机制来获取到需要转换的类型,向JSONConfig类中注册。
在这个过程中,封装了日期格式的传递,方便大家得到自己想要的日期格式。