Jackson序列化(5) — Jackson的ObjectMapper.DefaultTyping.NON_FINAL属性

Jackson序列化(1)— [SpringBoot2.x]-Jackson在HttpMessageConverter(消息转换器)中的使用
Jackson序列化(2)— [SpringBoot2.x]-Spring容器中ObjectMapper配置
Jackson序列化(3)— Jackson中ObjectMapper配置详解
Jackson序列化(4)— Jackson“默认的”时间格式化类—StdDateFormat解析
Jackson序列化(5) — Jackson的ObjectMapper.DefaultTyping.NON_FINAL属性
Jackson序列化(6)— Java使用Jackson进行序列化

ObjectMapperjackson的核心。Jackson的Json操作都是在ObjectMapper中实现的。ObjectMapper有一个配置:

ObjectMapper om = new ObjectMapper();
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);

这个配置会对JSON的序列化和反序列化带来什么影响呢?

在SpringBoot2.x整合Redis时,若使用Jackson作为序列化工具,为何要设置该配置?

1. 源码的描述

源码:com.fasterxml.jackson.databind.ObjectMapper.DefaultTyping

        /**
         * Value that means that default typing will be used for
         * all non-final types, with exception of small number of
         * "natural" types (String, Boolean, Integer, Double), which
         * can be correctly inferred from JSON; as well as for
         * all arrays of non-final types.
         *<p>
         * Since 2.4, this does NOT apply to {@link TreeNode} and its subtypes.
         */
        NON_FINAL

源码的解释是:对于除了一些自然类型(String、Double、Integer、Double)类型外的非常量(non-final)类型,类型将会用在值的含义上。以便可以在JSON串中正确的推测出值所属的类型。

2. 区别

1. 序列化的区别

我们先不设置ObjectMapper.DefaultTyping.NON_FINAL属性。对POJO进行序列化。

    @Test
    public void testDefaultTyping() throws JsonProcessingException {
        ObjectMapper objectMapper = new ObjectMapper();
        //POJO无public的属性或方法时,不报错
        objectMapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS);
        //null值字段不显示
        objectMapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
        //美化JSON输出
        objectMapper.enable(SerializationFeature.INDENT_OUTPUT);
//        objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        User user = new User();
        user.setAa(new Aa());   //普通POJO对象
        user.setAmount1(1.1);   //Double对象
        user.setUName("Tom");  //String类型
        user.setDate(new Date()); //Date类型
        String string = objectMapper.writeValueAsString(user);   //解析对象
        System.out.println(string);
    }

输出结果如图1所示,这个是正常的JSON串。

图1_输出结果.png

接着开启objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);配置,对POJO对象进行序列化。得到的JSON串的值中带有对象的类型。这也就是源码的含义。如图2所示。

图2_开启DefaultTyping.NON_FINAL的输出配置.png

2. 反序列化的区别

我们将POJO对象进行序列化,大多数是为了存储或传输。最终我们还是需要进行反序列化成为POJO的。那么这两种JSON串进行反序列化时会有问题吗?

我们对ObjectMapper设置了objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);参数,对普通的JSON串进行反序列化。

    @Test
    public void testDefaultTyping() throws IOException {
        ObjectMapper objectMapper = new ObjectMapper();
        //POJO无public的属性或方法时,不报错
        objectMapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS);
        //null值字段不显示
        objectMapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
        //序列化JSON串时,在值上打印出对象类型
        objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        String string="{\"date\":1571887801613,\"aa\":{},\"amount1\":1.1,\"uname\":\"Tom\"}";
        User user = objectMapper.readValue(string, User.class);
        System.out.println(user);
    }

运行结果,如下列代码所示,因为我们的JSON串中不含有字段的类型,所以不能进行反序列化。

com.fasterxml.jackson.databind.exc.MismatchedInputException: 
Unexpected token (START_OBJECT), expected START_ARRAY: 
need JSON Array to contain As.WRAPPER_ARRAY type information for class com.galax.jackson.User
at [Source: (String)"{"date":1571887801613,"aa":{},"amount1":1.1,"uname":"Tom"}"; line: 1, column: 1]

未设置objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);字段,对含有值类型的JSON串进行反序列化。

    @Test
    public void testDefaultTyping() throws IOException {
        ObjectMapper objectMapper = new ObjectMapper();
        //POJO无public的属性或方法时,不报错
        objectMapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS);
        //null值字段不显示
        objectMapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
        //序列化JSON串时,在值上打印出对象类型
//        objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        String string="[\"com.galax.jackson.User\",{\"date\":[\"java.util.Date\",1571888059184],\"aa\":[\"com.galax.jackson.Aa\",{}],\"amount1\":1.1,\"uname\":\"Tom\"}]";
        User user = objectMapper.readValue(string, User.class);
        System.out.println(user);
    }

运行结果依旧是不能解析。

com.fasterxml.jackson.databind.exc.MismatchedInputException: 
Cannot deserialize instance of `com.galax.jackson.User` out of START_ARRAY token

3. 注意事项

Spring源码中是使用容器中的ObjectMapper对象进行序列化和反序列化。当我们将自定义的ObjectMapper对象放入IOC容器中后,会自动覆盖SpringBoot自动装载的ObjectMapper对象。若是我们在自定义的ObjectMapper中设置了objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);属性。那么可能会影响我们的@RequestBody反序列化JSON串,导致Spring Boot默认错误返回格式变成数组@RequestBody无法解析Json格式异常。

如何去自定义配置IOC容器中的ObjectMapper对象,实际上,SpringBoot给了两种方式,详情请参考:SpringBoot配置ObjectMapper源码,以及ObjectMapper序列化/反序列化配置

4. 项目运用

SpringBoot2.X整合Redis缓存中,使用Jackson作为序列化工具,其中就使用了ObjectMapper.DefaultTyping.NON_FINAL配置。

        Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
        ObjectMapper om = new ObjectMapper();
        om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
       //序列化时允许非常量字段均输出类型
        om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        jackson2JsonRedisSerializer.setObjectMapper(om);

因为数据在反序列化为具体类时,需要传入具体的对象类型。而Redis的序列化配置是公共配置。我们只能传入Object.class类型进行反序列化。

  • 序列化得到的字段为


    序列化对象格式.png
  • 反序列化会得到LinkedHashMap类型,该类型不能强转为具体的对象类型,即抛出

java.lang.ClassCastException: java.util.LinkedHashMap cannot be cast to com.galax.pojo.Account
  • 所以我们需要ObjectMapper.DefaultTyping.NON_FINAL配置,在序列化时记录对象类型,以便反序列化时得到对应的具体对象。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335