作用一种简单方便的数据传输方案,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
- IsoDateTimeConverter convert = new IsoDateTimeConverter();
- string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);
- IsoDateTimeConverter convert = new IsoDateTimeConverter();
- string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);
使用这种方式格式化后的结果是
"2013-05-31T15:14:13.1294788+08:00"
b.
JavaScriptDateTimeConverter
- JavaScriptDateTimeConverter convert = new JavaScriptDateTimeConverter();
- string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);
- JavaScriptDateTimeConverter convert = new JavaScriptDateTimeConverter();
- string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);
使用这种方式格式化后的结果是
new
Date(1369984667554)
然而,即使如此,C#的三种格式没有一种是与Java相同的,仍然无法正确对接。
2、解决方案 经过反复测试,最终决定两端均使用
"yyyy-MM-dd HH:mm:ss.SSS"
的格式进行传输,这也是到目前为止测试成功的唯一一种方案。Java端:
- JsonGenerator jsonGenerator = null;
- ObjectMapper objectMapper = null;
-
- objectMapper = new ObjectMapper();
- SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS");
- objectMapper.getDeserializationConfig().setDateFormat(formatter);
-
- try {
- jsonGenerator = objectMapper.getJsonFactory().createJsonGenerator(System.out, JsonEncoding.UTF8);
- } catch (IOException e) {
- e.printStackTrace();
- }
- String json = "{\"name\":\"YK\",\"value\":3,\"tm\":\"2013-05-31 02:53:20.123\"}";
- try {
- MyBean b = objectMapper.readValue(json, MyBean.class);
- System.out.println(b.getTm());
- } catch (Exception e) {
- e.printStackTrace();
- }
- JsonGenerator jsonGenerator = null;
- ObjectMapper objectMapper = null;
-
- objectMapper = new ObjectMapper();
- SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS");
- objectMapper.getDeserializationConfig().setDateFormat(formatter);
-
- try {
- jsonGenerator = objectMapper.getJsonFactory().createJsonGenerator(System.out, JsonEncoding.UTF8);
- } catch (IOException e) {
- e.printStackTrace();
- }
- String json = "{\"name\":\"YK\",\"value\":3,\"tm\":\"2013-05-31 02:53:20.123\"}";
- try {
- MyBean b = objectMapper.readValue(json, MyBean.class);
- System.out.println(b.getTm());
- } catch (Exception e) {
- e.printStackTrace();
- }
- IsoDateTimeConverter convert = new IsoDateTimeConverter();
- convert.DateTimeFormat = "yyyy-MM-dd HH:mm:ss.fff";
- string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);
- IsoDateTimeConverter convert = new IsoDateTimeConverter();
- convert.DateTimeFormat = "yyyy-MM-dd HH:mm:ss.fff";
- string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);
文中用的到MyBean类型定义:
- import java.sql.Timestamp;
-
- public class MyBean {
-
- private String name;
- private String value;
- private Timestamp tm;
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String getValue() {
- return value;
- }
- public void setValue(String value) {
- this.value = value;
- }
- public Timestamp getTm() {
- return tm;
- }
- public void setTm(Timestamp tm) {
- this.tm = tm;
- }
-
- }
- import java.sql.Timestamp;
-
- public class MyBean {
-
- private String name;
- private String value;
- private Timestamp tm;
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String getValue() {
- return value;
- }
- public void setValue(String value) {
- this.value = value;
- }
- public Timestamp getTm() {
- return tm;
- }
- public void setTm(Timestamp tm) {
- this.tm = tm;
- }
-
- }
后记:
我使用的是比较老的.net
2.0,而C#的JSON转换在高版本中是提供原生支持的,此问题在4.0/4.5的Framework中是否有更好的解决方案,期待大家反馈。
另外,C#向Java传递数据时,其生成的Json中往往包含多余的字段,在服务器端解析时可能报错,解决方法如下: