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

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

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

[b]1、基本情况[/b]

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

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

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

a. IsoDateTimeConverter[/size]


IsoDateTimeConverter convert = new IsoDateTimeConverter();
string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);


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

b. JavaScriptDateTimeConverter[/size]


JavaScriptDateTimeConverter convert = new JavaScriptDateTimeConverter();
string ret = JsonConvert.SerializeObject(bean, Formatting.None, convert);


[size=medium]使用这种方式格式化后的结果是
new Date(1369984667554)

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

[b]2、解决方案[/b]

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

Java端:[/size]


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);


[size=medium]文中用的到MyBean类型定义:[/size]

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;
}

}


[size=medium]后记:

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

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


objectMapper.getDeserializationConfig().set( org.codehaus.jackson.map.DeserializationConfig.Feature.FAIL_ON_UNKNOWN_PROPERTIES, false);


[size=medium]感谢 elgs 和 bypasscc 对本文的支持。[/size]
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
RSA加密算法是一种非对称加密算法,公钥加密,私钥解密。在C#Java中,都内置了RSA加密算法的实现,但是在使用过程中,由于两个语言的实现细节不同,会导致在C#Java进行RSA加密的时候出现格式不兼容的问题。 为了解决这个问题,我们可以使用以下方法将C#格式的RSA公钥/私钥转换为Java格式的RSA公钥/私钥: 1. 将C#的RSA公钥/私钥转换为XML格式的字符串。 C#中,可以使用以下方法将RSA公钥/私钥转换为XML格式的字符串: ```csharp string publicKeyXml = rsa.ToXmlString(false); // 公钥 string privateKeyXml = rsa.ToXmlString(true); // 私钥 ``` 2. 将XML格式的字符串转换为Java中的BigInteger类型。 Java中,可以使用以下方法将XML格式的字符串转换为BigInteger类型: ```java BigInteger modulus = new BigInteger(1, Base64.getDecoder().decode(modulusString)); BigInteger exponent = new BigInteger(1, Base64.getDecoder().decode(exponentString)); ``` 其中,modulusString和exponentString分别是XML格式的字符串中的Modulus和Exponent节点的值,使用Base64解码后转为BigInteger类型。 3. 使用BigInteger类型创建Java中的RSAPublicKey/RSAPrivateKey对象。 Java中,可以使用以下方法创建RSAPublicKey/RSAPrivateKey对象: ```java RSAPublicKeySpec publicKeySpec = new RSAPublicKeySpec(modulus, exponent); RSAPublicKey publicKey = (RSAPublicKey) KeyFactory.getInstance("RSA").generatePublic(publicKeySpec); RSAPrivateKeySpec privateKeySpec = new RSAPrivateKeySpec(modulus, exponent); RSAPrivateKey privateKey = (RSAPrivateKey) KeyFactory.getInstance("RSA").generatePrivate(privateKeySpec); ``` 其中,modulus和exponent是从XML格式的字符串中解析得到的BigInteger类型的值。 通过以上步骤,我们就可以将C#格式的RSA公钥/私钥转换为Java格式的RSA公钥/私钥了。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值