Gson对Map中值为null的字段转换为空字符串输出-阿里云开发者社区

开发者社区> Java实用技术@Pandas> 正文

Gson对Map中值为null的字段转换为空字符串输出

简介: 之前我写过一篇文章《Gson对字符串null的字段转换为空字符串输出》,有个兄弟评论说:定义返回的对象,code,msg,object data类型 data类型里面如果是List的map好像还是转不了。这次写一下关于Gson对Map中值为null的字段转换为空字符串输出。
+关注继续查看


问题背景

之前我写过一篇文章《Gson对字符串null的字段转换为空字符串输出》,有个兄弟评论说:定义返回的对象,code,msg,object data类型 data类型里面如果是List的map好像还是转不了。。

image-20211020230606185.png

上图代码的maps输出结果是:[{"id":"123"},{"id":"123"},{"id":"123"}]

看了上面的代码,不知道其他同学有什么想法?

我发现还是有人没有理解原理,如果不知道为什么,这时候debug源码很快就可以获得答案。

解析GSON适配器

Gson有自己适配器,所有map都会走默认的MapTypeAdapterFactory类。

在这个类里面write方法中,对value的判断调用valueTypeAdapter.write(out, entry.getValue());方法。代码如下:

@Override 
public void write(JsonWriter out, Map<K, V> map) throws IOException {
      if (map == null) {
        out.nullValue();
        return;
      }
      if (!complexMapKeySerialization) {
        out.beginObject();
        for (Map.Entry<K, V> entry : map.entrySet()) {
          out.name(String.valueOf(entry.getKey()));
          valueTypeAdapter.write(out, entry.getValue());
        }
        out.endObject();
        return;
      }
      //...
}

因为map的泛型在开始时并不知道key和value的类型是什么,所以在MapTypeAdapterFactory类初始化的时候,valueAdapter实际上是ObjectTypeAdapter方法。我们看下具体write方法:

@Override 
public void write(JsonWriter out, Object value) throws IOException {
    if (value == null) {
      out.nullValue();
      return;
    }
    TypeAdapter<Object> typeAdapter = (TypeAdapter<Object>) gson.getAdapter(value.getClass());
    if (typeAdapter instanceof ObjectTypeAdapter) {
      out.beginObject();
      out.endObject();
      return;
    }
    typeAdapter.write(out, value);
  }

因为value == null,所以最后调用JsonWriter.nullValue();

public JsonWriter nullValue() throws IOException {
    if (deferredName != null) {
      if (serializeNulls) {
        writeDeferredName();
      } else {
        deferredName = null;
        return this; // skip the name and the value
      }
    }
    beforeValue();
    out.write("null");
    return this;
  }

到这里大家就明白了吧,我们没有配置serializeNulls,这里的serializeNulls == false,所以后面分支的注释已经说了:如果对于null值,会把它的name置空,跳过,也就是最后不输出。

问题根因大家知道了,解决办法应该有好几个:

(1)最常用的就是自定义一个map的adapter,让Gson启动加载,上篇文章自定义的string的适配器不是map的适配器,所以是不起作用的。

(2)重写MapTypeAdapterFactory类。

(3)改写JsonWriter类。

因为Gson很多类都是final的,无法继承覆写,还是使用第一个比较简单。

下面是我写的自定义map适配器:

public class MyMapTypeAdapter<K, V> extends TypeAdapter<Map<K, V>>
{
    @Override
    public void write(JsonWriter out, Map<K, V> map) throws IOException
    {
        if (map == null)
        {
            out.nullValue();
            return;
        }
        out.beginObject();
        for (Map.Entry<K, V> entry : map.entrySet())
        {
            out.name(String.valueOf(entry.getKey()));
            // 核心代码在这里,如果value是null,输出空字符串
            Object value = (entry.getValue() == null) ? "" : entry.getValue();
            new Gson().getAdapter(Object.class).write(out, value);
        }
        out.endObject();
        return;
    }
    @Override
    public Map<K, V> read(JsonReader var1) throws IOException
    {
        // TODO Auto-generated method stub
        return null;
    }
}

运行结果[{"name":"","id":"123"},{"name":"","id":"123"},{"name":"","id":"123"}]

gson.jpg


到这里,就解决了Map的value中为空就不输出的问题。


思考

上面有个细节,就是为啥Gson不对ObjectTypeAdapter中的null值提供写为""的方法?

因为我们常说的null其实是一种像量子态的引用类型,你可以说它没有类型,但是你又可以把它转为任何类型。

Gson不知道你传给它的null到底是(String)null,还是(Integer)null,也许String类型的null想变成"",也许Integer类型的null就想变成0。所以自由权还是给使用者,自己定义吧。

如果有人使用阿里的fastjson,对于map的null值处理也要自定义filter才行,默认的SerializerFeature.WriteNullStringAsEmpty不行。

备注:本文使用的 Gson 版本为 gson-2.5


我是Pandas,专注Java编程实用技术分享,微信公众号:Java实用技术手册\

关注可了解更多java技能和互联网面试技巧。问题或建议,请公众号留言。\

如果你觉得这篇文章对你有帮助,欢迎一键三连

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
json字符串还原map
今天有个需求,由于调用第三方接口一直调不通,但是又需要先把接口写好供前端调用,于是想到以前写的一个json字符串还原的类。这里做个笔记。 package com.
975 0
51 Nod 1791 合法括号子段【分治+字符串】
1791 合法括号子段 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题 有一个括号序列,现在要计算一下它有多少非空子段是合法括号序列。 合法括号序列的定义是: 1.空序列是合法括号序列。
843 0
SAP LSMW 导入OPEN PO 单据时候’税码’字段的处理
SAP LSMW Standard Batch (Direct) Input 方式制作的LSMW工具导入OPEN PO 单据时候’税码’字段的处理 如下的Open PO 批量导入LSMW工具,   ...
1080 0
利用MaxCompute内建函数及UDTF转换json格式日志数据
本文介绍了如何使用MaxCompute UDF对JSON格式的日志进行信息提取和转换。
8159 0
4
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
《2021云上架构与运维峰会演讲合集》
立即下载
《零基础CSS入门教程》
立即下载
《零基础HTML入门教程》
立即下载