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());方法。代码如下:

@Overridepublicvoidwrite(JsonWriterout, Map<K, V>map) throwsIOException {
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方法:

@Overridepublicvoidwrite(JsonWriterout, Objectvalue) throwsIOException {
if (value==null) {
out.nullValue();
return;
    }
TypeAdapter<Object>typeAdapter= (TypeAdapter<Object>) gson.getAdapter(value.getClass());
if (typeAdapterinstanceofObjectTypeAdapter) {
out.beginObject();
out.endObject();
return;
    }
typeAdapter.write(out, value);
  }

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

publicJsonWriternullValue() throwsIOException {
if (deferredName!=null) {
if (serializeNulls) {
writeDeferredName();
      } else {
deferredName=null;
returnthis; // skip the name and the value      }
    }
beforeValue();
out.write("null");
returnthis;
  }

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

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

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

(2)重写MapTypeAdapterFactory类。

(3)改写JsonWriter类。

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

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

publicclassMyMapTypeAdapter<K, V>extendsTypeAdapter<Map<K, V>>{
@Overridepublicvoidwrite(JsonWriterout, Map<K, V>map) throwsIOException    {
if (map==null)
        {
out.nullValue();
return;
        }
out.beginObject();
for (Map.Entry<K, V>entry : map.entrySet())
        {
out.name(String.valueOf(entry.getKey()));
// 核心代码在这里,如果value是null,输出空字符串Objectvalue= (entry.getValue() ==null) ?"" : entry.getValue();
newGson().getAdapter(Object.class).write(out, value);
        }
out.endObject();
return;
    }
@OverridepublicMap<K, V>read(JsonReadervar1) throwsIOException    {
// TODO Auto-generated method stubreturnnull;
    }
}

运行结果[{"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技能和互联网面试技巧。问题或建议,请公众号留言。\

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

相关文章
|
2天前
|
SQL 数据库
MyBatisPlus-null判定及查询指定字段
MyBatisPlus-null判定及查询指定字段
280 0
|
2天前
|
Java
java Map删除值为null的元素
java Map删除值为null的元素
|
2天前
|
分布式计算 大数据 专有云
MaxCompute产品使用合集之需要将多个字段concat起来,有时候遇到null值,该怎么做
MaxCompute作为一款全面的大数据处理平台,广泛应用于各类大数据分析、数据挖掘、BI及机器学习场景。掌握其核心功能、熟练操作流程、遵循最佳实践,可以帮助用户高效、安全地管理和利用海量数据。以下是一个关于MaxCompute产品使用的合集,涵盖了其核心功能、应用场景、操作流程以及最佳实践等内容。
|
2天前
|
存储 关系型数据库 MySQL
Flink CDC中mysql cdc 抽取这个时间字段的值为null 有什么好的解决方案吗 ?
Flink CDC中mysql cdc 抽取这个时间字段的值为null 有什么好的解决方案吗 ?
108 0
|
2天前
|
XML Java 数据库连接
MyBatis返回Map时值为null的字段会丢失
MyBatis返回Map时值为null的字段会丢失
|
2天前
|
关系型数据库 MySQL
mysql中判断NULL和空字符串
mysql中判断NULL和空字符串
12 0
|
2天前
|
Java 数据库连接 mybatis
mybatis plus字段为null或空字符串把原来的数据也更新了,只需要注解
mybatis plus字段为null或空字符串把原来的数据也更新了,只需要注解
24 0
|
2天前
|
移动开发 关系型数据库 MySQL
mysql删除为NULL或者空字符串‘‘或者‘null’的或者删除空格的
mysql删除为NULL或者空字符串‘‘或者‘null’的或者删除空格的
15 1
|
2天前
|
SQL Java 关系型数据库
JDBC PreparedStatement 字段值为null导致TBase带宽飙升的案例分析
JDBC PreparedStatement 字段值为null导致TBase带宽飙升的案例分析
54 0
|
5月前
|
关系型数据库 MySQL 开发者
MySQL 字段约束 null, not null, default, auto_increment
前言:转载,觉得有用就发了一遍 在 MySQL 中,每个字段定义都包含附加约束或者修饰符,这些可以用来增加对所输入数据的约束。 今天我们就来看一下 MySQL 的字段约束: NULL 和 NOT NULL 修饰符、DEFAULT 修饰符,AUTO_INCREMENT 修饰符。
122 0