Lombok的踩坑系列之@Builder

背景:

Lombok 这个插件大家日常工作中几乎是必备的,几个简单的注解就可以帮助我们减少一大坨get/set方法等;其中@Builder注解使用的也很广泛,使用了建造者模式帮助我们构建出个性化的对象,本次踩坑点就在这个地方。

先讲一下踩坑的大致流程,在一个需求中需要对接口内部的一个上下文对象 增加一个属性Map,而这个上下文对象在别的接口中也有使用,那就需要兼容其他接口,所以我给这个新增的Map属性增加一个默认值 Map<Object,Object> map = Maps.newHashMap() ,然而还是获取这个属性的时候发生了异常,原因就在当前类上面的@Builder注解,下文会举一个例子具体说明一下。

举例:

package com.shizhuang.duapp.nbinterface.interfaces.facade;

import com.google.common.collect.Maps;
import lombok.Builder;
import lombok.Getter;

import java.util.Map;

@Getter
@Builder
public class CommodityModel {

    private String title;

    private Long brandId;

    private Integer channelCode;

    private Map<String,Long> extraInfoMap = Maps.newHashMap();

    public static void main(String[] args) {

        CommodityModel model = CommodityModel.builder()
                .brandId(100L)
                .title("NB 新百伦")
                .build();
        Object price = model.getExtraInfoMap().getOrDefault("price", 100L);
        System.out.println("price:" + price);
    }
}

问题来了,如上代码直接执行main方法,是否会打印出 "price: 100" ?

答案分割图

嗯哼,答案是大家贼熟悉的 NPE

看到这NPE,肯定是 extraInfoMap 这个属性是Null ,但是我们明明给了一个默认值嘛,为啥子会是Null 呢?答案就在编译后的代码中,如下:(着重关注标记的代码)

java 复制代码
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by FernFlower decompiler)
//
package com.shizhuang.duapp.nbinterface.interfaces.facade;

import com.google.common.collect.Maps;
import java.util.Map;

public class CommodityModel {
    private String title;
    private Long brandId;
    private Integer channelCode;
    private Map<String, Long> extraInfoMap = Maps.newHashMap();

    public static void main(String[] args) {
        CommodityModel model = builder().brandId(100L).title("NB 新百伦").build();
        Object price = model.getExtraInfoMap().getOrDefault("price", 100L);
        System.out.println("price:" + price);
    }

    CommodityModel(String title, Long brandId, Integer channelCode, Map<String, Long> extraInfoMap) {
        this.title = title;
        this.brandId = brandId;
        this.channelCode = channelCode;
        this.extraInfoMap = extraInfoMap;
    }
    
    //方法1
    public static CommodityModelBuilder builder() {
        return new CommodityModelBuilder();
    }

  --- 省略Get方法 ---

    public static class CommodityModelBuilder {
        private String title;
        private Long brandId;
        private Integer channelCode;
        private Map<String, Long> extraInfoMap;

        CommodityModelBuilder() {
        }
    //方法2
        public CommodityModelBuilder title(String title) {
            this.title = title;
            return this;
        }
    //方法3
        public CommodityModelBuilder brandId(Long brandId) {
            this.brandId = brandId;
            return this;
        }

        public CommodityModelBuilder channelCode(Integer channelCode) {
            this.channelCode = channelCode;
            return this;
        }

        public CommodityModelBuilder extraInfoMap(Map<String, Long> extraInfoMap) {
            this.extraInfoMap = extraInfoMap;
            return this;
        }
    //方法4
        public CommodityModel build() {
            return new CommodityModel(this.title, this.brandId, this.channelCode, this.extraInfoMap);
        }

        public String toString() {
            return "CommodityModel.CommodityModelBuilder(title=" + this.title + ", brandId=" + this.brandId + ", channelCode=" + this.channelCode + ", extraInfoMap=" + this.extraInfoMap + ")";
        }
    }
}

Lombok的@Builder 注解在编译期间会帮我们生成一个内部的Builder类,并生成一个创建这个内部builder对象的静态方法(方法1),然后我们的代码是调用了方法1,方法2,方法3和方法4,其中方法4中的this.extraInfoMap是内部类中的属性并没有默认值,所以build()方法返回的对象extraInfoMap就是一个null;

解决:

在需要默认值的属性上面增加 @Builder.Default 注解

java 复制代码
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by FernFlower decompiler)
//

package com.shizhuang.duapp.nbinterface.interfaces.facade;

import com.google.common.collect.Maps;
import java.util.Map;

public class CommodityModel {
    private String title;
    private Long brandId;
    private Integer channelCode;
    private Map<String, Long> extraInfoMap;
  
    // 方法1
    private static Map<String, Long> $default$extraInfoMap() {
        return Maps.newHashMap();
    }
     --- 省略部分代码 ---

    public Map<String, Long> getExtraInfoMap() {
        return this.extraInfoMap;
    }

    public static class CommodityModelBuilder {
        private String title;
        private Long brandId;
        private Integer channelCode;
        private boolean extraInfoMap$set;
        private Map<String, Long> extraInfoMap$value;

        CommodityModelBuilder() {
        }

        public CommodityModelBuilder title(String title) {
            this.title = title;
            return this;
        }
        
        --- 省略部分代码 ---

        public CommodityModelBuilder extraInfoMap(Map<String, Long> extraInfoMap) {
            this.extraInfoMap$value = extraInfoMap;
            // 标记用户已对目标属性赋值处理了
            this.extraInfoMap$set = true;
            return this;
        }

        public CommodityModel build() {
            // this.extraInfoMap$value 是内部类的属性
            Map<String, Long> extraInfoMap$value = this.extraInfoMap$value;
            // 用户如果没有操作,则使用方法1为内部类赋值
            if (!this.extraInfoMap$set) {
                extraInfoMap$value = CommodityModel.$default$extraInfoMap();
            }
            // 使用内部类的属性创建对象
            return new CommodityModel(this.title, this.brandId, this.channelCode, extraInfoMap$value);
        }
    }
}

此时再看编译后的代码,会发现内部类中有一个属性extraInfoMap$set 会标记用户是否对extraInfoMap属性处理过,没有操作的话就会赋值我们加的默认值 = Maps.newHashMap();

总结:

日常我们业务开发中有很多小的需求,只需要增加一个属性就可以解决,此时就要注意历史逻辑中是否用 Lombok 的 Builder方式创建对象,

相关推荐
李先静几秒前
AWTK-WEB 快速入门(1) - C 语言应用程序
c语言·开发语言·前端
weixin_4467077414 分钟前
IDEA2024 maven构建跳过测试
java·maven
开朗觉觉21 分钟前
RabbitMQ高可用&&延迟消息&&惰性队列
java·rabbitmq·java-rabbitmq
zmd-zk22 分钟前
flink学习(3)——方法的使用—对流的处理(map,flatMap,filter)
java·大数据·开发语言·学习·flink·tensorflow
昵称202125 分钟前
flink1.16+连接Elasticsearch7官方例子报错解决方案
java·flink·es7
圆蛤镇程序猿26 分钟前
【什么是SpringMVC】
开发语言
爱编程的小生37 分钟前
Easyexcel(6-单元格合并)
java·excel
Domain-zhuo44 分钟前
JS对于数组去重都有哪些方法?
开发语言·前端·javascript
小白不太白9501 小时前
设计模式之 迭代器模式
java·设计模式·迭代器模式
闲人一枚(学习中)1 小时前
设计模式-创建型-单例模式
java·单例模式·设计模式