首页 > 代码库 > Java与C#间json日期格式互转完美解决方案

Java与C#间json日期格式互转完美解决方案

http://blog.csdn.net/wilsonke/article/details/24362851

 
作用一种简单方便的数据传输方案,JSON已经成为替代XML的事实标准。然而在JSON中,时间(DateTime,Timestamp,Date等)格式一直没有很好地统一,当需要跨平台序列化/反序列化时,遇到不少麻烦。作者经过反复尝试,解决了C#与Java通过JSON进行时间传输的困难。

C#解析Java/Javascript生成的JSON并不困难,但Java解析C#生成的JSON困难重重。下面就此问题重点介绍。 

1、基本情况 

Java端: 
Java端常用的json-lib库不支持Timestamp类型的反序列化(有人说可以改数据类型啊,对不起,很多都是历史代码不是想改就能改的)。而Jackson与gson能支持毫秒数形式的反序列化。应该说,Java的序列化行为是比较简单、符合常理的。

C#端: 
默认情况下,C#的时间将被格式化为 "/Date(1294499956278+0800)/" 这种形式。很显然,这种形式难以处理。 

幸好,JsonConvert第三方库提供了两种额外的格式: 

a. IsoDateTimeConverter 

C#代码 复制代码 收藏代码
  1. IsoDateTimeConverter convert = new IsoDateTimeConverter();  
  2. string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);  
[C#] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. IsoDateTimeConverter convert = new IsoDateTimeConverter();  
  2. string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);  


使用这种方式格式化后的结果是 
"2013-05-31T15:14:13.1294788+08:00" 

b. JavaScriptDateTimeConverter 

C#代码 复制代码 收藏代码
  1. JavaScriptDateTimeConverter convert = new JavaScriptDateTimeConverter();  
  2. string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);  
[C#] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. JavaScriptDateTimeConverter convert = new JavaScriptDateTimeConverter();  
  2. string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);  


使用这种方式格式化后的结果是 
new Date(1369984667554) 

然而,即使如此,C#的三种格式没有一种是与Java相同的,仍然无法正确对接。 

2、解决方案 

经过反复测试,最终决定两端均使用 "yyyy-MM-dd HH:mm:ss.SSS" 的格式进行传输,这也是到目前为止测试成功的唯一一种方案。

Java端: 

Java代码 复制代码 收藏代码
  1. JsonGenerator jsonGenerator = null;  
  2. ObjectMapper objectMapper = null;  
  3.           
  4. objectMapper = new ObjectMapper();  
  5. SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS");   
  6. objectMapper.getDeserializationConfig().setDateFormat(formatter);  
  7.           
  8. try {  
  9.     jsonGenerator = objectMapper.getJsonFactory().createJsonGenerator(System.out, JsonEncoding.UTF8);  
  10. catch (IOException e) {  
  11.     e.printStackTrace();  
  12. }  
  13. String json = "{\"name\":\"YK\",\"value\":3,\"tm\":\"2013-05-31 02:53:20.123\"}";  
  14. try {  
  15.     MyBean b = objectMapper.readValue(json, MyBean.class);  
  16.     System.out.println(b.getTm());  
  17. catch (Exception e) {  
  18.     e.printStackTrace();  
  19. }  
[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. JsonGenerator jsonGenerator = null;  
  2. ObjectMapper objectMapper = null;  
  3.           
  4. objectMapper = new ObjectMapper();  
  5. SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS");   
  6. objectMapper.getDeserializationConfig().setDateFormat(formatter);  
  7.           
  8. try {  
  9.     jsonGenerator = objectMapper.getJsonFactory().createJsonGenerator(System.out, JsonEncoding.UTF8);  
  10. catch (IOException e) {  
  11.     e.printStackTrace();  
  12. }  
  13. String json = "{\"name\":\"YK\",\"value\":3,\"tm\":\"2013-05-31 02:53:20.123\"}";  
  14. try {  
  15.     MyBean b = objectMapper.readValue(json, MyBean.class);  
  16.     System.out.println(b.getTm());  
  17. catch (Exception e) {  
  18.     e.printStackTrace();  
  19. }  


C#代码 复制代码 收藏代码
  1. IsoDateTimeConverter convert = new IsoDateTimeConverter();  
  2. convert.DateTimeFormat = "yyyy-MM-dd HH:mm:ss.fff";  
  3. string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);  
[C#] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. IsoDateTimeConverter convert = new IsoDateTimeConverter();  
  2. convert.DateTimeFormat = "yyyy-MM-dd HH:mm:ss.fff";  
  3. string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);  


文中用的到MyBean类型定义: 
Java代码 复制代码 收藏代码
  1. import java.sql.Timestamp;  
  2.   
  3. public class MyBean {  
  4.   
  5.     private String name;  
  6.     private String value;  
  7.     private Timestamp tm;  
  8.     public String getName() {  
  9.         return name;  
  10.     }  
  11.     public void setName(String name) {  
  12.         this.name = name;  
  13.     }  
  14.     public String getValue() {  
  15.         return value;  
  16.     }  
  17.     public void setValue(String value) {  
  18.         this.value = value;  
  19.     }  
  20.     public Timestamp getTm() {  
  21.         return tm;  
  22.     }  
  23.     public void setTm(Timestamp tm) {  
  24.         this.tm = tm;  
  25.     }  
  26.       
  27. }  
[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. import java.sql.Timestamp;  
  2.   
  3. public class MyBean {  
  4.   
  5.     private String name;  
  6.     private String value;  
  7.     private Timestamp tm;  
  8.     public String getName() {  
  9.         return name;  
  10.     }  
  11.     public void setName(String name) {  
  12.         this.name = name;  
  13.     }  
  14.     public String getValue() {  
  15.         return value;  
  16.     }  
  17.     public void setValue(String value) {  
  18.         this.value = value;  
  19.     }  
  20.     public Timestamp getTm() {  
  21.         return tm;  
  22.     }  
  23.     public void setTm(Timestamp tm) {  
  24.         this.tm = tm;  
  25.     }  
  26.       
  27. }  


后记: 

我使用的是比较老的.net 2.0,而C#的JSON转换在高版本中是提供原生支持的,此问题在4.0/4.5的Framework中是否有更好的解决方案,期待大家反馈。

另外,C#向Java传递数据时,其生成的Json中往往包含多余的字段,在服务器端解析时可能报错,解决方法如下: