关于Java和.NET之间的通信问题(JSON)

版权声明:此文章转载自博客园

如需转载请联系听云College团队成员阮小乙,邮箱:ruanqy#tingyun.com

前言:

最近项目在某XX领导的所谓指引下,非要转型Java,转就转吧,在转的过程前期是个痛苦期,特别.NET旧有项目和Java新项目需要通信时。

进入主题,Java和.NET之间需要通信,这时媒介很多,协议很多。其中json应该是比较常用的。而json一开始看过去,觉得这个问题实在简单要弱智,没什么好深究的。但是你实际做的过程中却不是这样。

比如Java序列化/反序列化时区默认不是取服务器上的时区,比如.NET默认序列化DateTime格式是微软自家定义的,到了Java那边自然转不了,报错。

先出点会报错的代码看看:

ObjectMapper mapper = new ObjectMapper();
JavaType javaType = mapper.getTypeFactory().constructParametricType(ArrayList.class, Message.class);
List<Message> olds = mapper.readValue(sv,javaType);

一般而言,转换一个列表Java使用spring集成的jackson语法如上,如果是Java自己序列化的数据呢,是反序列化回来是问题的,但是.NET那边序列化的数据呢?不好意思,时间格式转换失败,不好意思,bool类型转换失败。比如以下错误信息:

com.fasterxml.jackson.databind.exc.InvalidFormatException: Can not construct instance of java.util.Date from String value '

.NET那边如果是Java序列化的数据呢?平常我一般都是用Newtonsoft序列化进去,也不用管太多其他。确实,不考虑Java方面,确实两个语言上json没什么好纠结的。但是如果现在还是和以前一样过来的数据,.NET bool类型会转换true/false,这时候Java是不能识别的。我们需要加一个转换器,下面看看怎么搞才能全局上支持:

public class BoolConverter : JsonConverter
    {

        public override bool CanConvert(Type objectType)
        {
            return true;
        }

        public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
        {
            try
            {
                
                if (reader.TokenType == JsonToken.Integer)
                {
                    //数值
                    return Convert.ToInt32(reader.Value) == 1;
                }
            }
             catch (Exception ex)
            {
                throw new Exception(string.Format("Error converting value {0} to type '{1}'", reader.Value, objectType));
            }
            throw new Exception(string.Format("Unexpected token {0} when parsing enum", reader.TokenType));
        }


        public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
        {
            if (value == null)
            {
                writer.WriteNull();
                return;
            }
            //bool.TryParse()
            bool bValue = (bool)value;

            if (bValue)
                writer.WriteValue(1);
            else
                writer.WriteValue(0);
        }
    }

加了转换器,这时候确实可以解决bool类型转换时多语言问题,通用的一般都是0,1。

再看看时间格式方面,你也需要做些时区以及格式上的设置:

.net:

var serializerSettings = new JsonSerializerSettings();
serializerSettings.DateFormatHandling = DateFormatHandling.IsoDateFormat;
serializerSettings.DateTimeZoneHandling = DateTimeZoneHandling.Utc;
return JsonConvert.SerializeObject(o, serializerSettings);

java:

mapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss'Z'");
df.setTimeZone(TimeZone.getTimeZone("GMT+8"));
mapper.getDeserializationConfig().with(df);
mapper.setDateFormat(df);

好了。

1、现在都在北京时间频道下,现在都是iso标准格式。

2、bool类型全部用0,1

其他

1、spring附带了jackson,所以没考虑用其他json,为了统一。

2、这类格式和类型问题可以做一个通用的拦截器,全局帮大伙做了。好了,去做全局转换拦截器了。

想阅读更多技术文章,请访问听云技术博客,访问听云官方网站感受更多应用性能优化魔力。

关于作者

阮小乙

人生就该充满正能量!

我要评论

评论请先登录,或注册