解决Lombok使用@Builder无法build父类属性的问题

目录
  • Lombok使用@Builder无法build父类属性
    • 问题描述
    • 解决方案
    • 使用示例
  • lombok @Builder注解和build父类属性问题
    • 1、简介
    • 2、使用
    • 3、@Builder注解对类做了什么?
    • 4、优缺点
    • 5、问题:@Builder注解不能 build 父类属性

Lombok使用@Builder无法build父类属性

问题描述

实体类使用Lombok的@Builder来实现Builder模式,但是如果使用了extend继承,则子类无法通过Builder来Build父类属性值

解决方案

父类增加@NoArgsConstructor、@AllArgsConstructor注解来增加无参和全参构造函数【注:父类属性不能设置为private,否则仍然无法访问,父类不允许有@Builder注解,否则会和子类冲突】

子类增加@NoArgsConstructor来增加无参构造函数,自定义一个全参构造函数(包含父类属性)

使用示例

父类:

package com.baijia.uqun.individual.management.facade;
import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.NoArgsConstructor;
import lombok.Setter;
/**
 * @Author C.W
 * @Date 2020/6/15 5:25 下午
 * @Description 父类
 */
@Setter
@Getter
@NoArgsConstructor
@AllArgsConstructor
public class Father {
    /**
     * 父类名称
     */
    public String fatherName;
}

子类:

package com.baijia.uqun.individual.management.facade;
import lombok.Builder;
import lombok.Getter;
import lombok.NoArgsConstructor;
import lombok.Setter;
/**
 * @Author C.W
 * @Date 2020/6/15 5:26 下午
 * @Description 子类
 */
@Setter
@Getter
@NoArgsConstructor
public class Child extends Father {
    /**
     * 子类名称
     */
    private String childName;
    /**
     * 用于解决Lombok的Builder无法Build父类属性问题
     *
     * @param fatherName
     * @param childName
     */
    @Builder(toBuilder = true)
    public Child(String fatherName, String childName) {
        super(fatherName);
        this.childName = childName;
    }
}

使用:

package com.baijia.uqun.individual.management.facade;
/**
 * @Author C.W
 * @Date 2020/6/15 5:28 下午
 * @Description 测试代码
 */
public class Test {
    public static void main(String[] args) {
        // 创建一个子类
        Child child = Child.builder()
                .fatherName("父类名称")
                .childName("子类名称")
                .build();
        System.out.println(String.format("父类名称:%s,子类名称:%s", child.getFatherName(), child.getChildName()));
    }
}

输出结果:

父类名称:父类名称,子类名称:子类名称

lombok @Builder注解和build父类属性问题

1、简介

通过@Builder注解,lombok可以方便的实践建造者模式。

2、使用

1)创建基类User

import lombok.*;
/**
 * @author Saint
 */
@Getter
@Setter
@ToString
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class User {
    private Long id;
    private String name;
}

2)创建子类UserExt

import lombok.*;
/**
 * @author Saint
 */
@Getter
@Setter
@Builder
@NoArgsConstructor
@AllArgsConstructor
@ToString
public class UserExt extends User{
    private String address;
    private Integer age;
}

3)分别使用Builder创建父类和子类

// 父类
User user = User.builder().id(1L).name("saint").build();
// 子类
UserExt userExt = UserExt.builder().address("南京").age(18).build();

3、@Builder注解对类做了什么?

反编译后的UserExt.class:

public class UserExt extends User {
    private String address;
    private Integer age;
    public static UserExt.UserExtBuilder builder() {
        return new UserExt.UserExtBuilder();
    }
    public String getAddress() {
        return this.address;
    }
    public Integer getAge() {
        return this.age;
    }
    public void setAddress(final String address) {
        this.address = address;
    }
    public void setAge(final Integer age) {
        this.age = age;
    }
    public UserExt() {
    }
    public UserExt(final String address, final Integer age) {
        this.address = address;
        this.age = age;
    }
    public String toString() {
        return "UserExt(address=" + this.getAddress() + ", age=" + this.getAge() + ")";
    }
    public static class UserExtBuilder {
        private String address;
        private Integer age;
        UserExtBuilder() {
        }
        public UserExt.UserExtBuilder address(final String address) {
            this.address = address;
            return this;
        }
        public UserExt.UserExtBuilder age(final Integer age) {
            this.age = age;
            return this;
        }
        public UserExt build() {
            return new UserExt(this.address, this.age);
        }
        public String toString() {
            return "UserExt.UserExtBuilder(address=" + this.address + ", age=" + this.age + ")";
        }
    }
}

注解在编译后使UserExt类中多了一个名为UserExt.UserExtBuilder的静态内部类。这个静态类拥有和UserExt类相同的属性,并且他额外实现了一些方法:

1.address、age的属性方法

其实这些方法和setAttribute十分类似,只是额外返回了实例本身,这使得它可以使用类似于链式调用的写法。

2.build方法

该方法调用UserExt类的全参构造方法来生成UserExt实例。

UserExt类还是实现了builder方法,这个方法生成一个空的UserExt.UserExtBuilder实例。

4、优缺点

  • 优点:

写法更优雅,不需要太多的set方法设置属性。

  • 缺点:

在生成UserExt实例之前,先创建了一个UserExt.UserExtBuilder实例,其占用了额外的内存。并且Java是按值传递,我们可以直接修改引用对象,不用新建一个对象再赋值;而Builder.build()方法每次调用都会new一个实例出来。

5、问题:@Builder注解不能 build 父类属性

从反编译后的UserExt类可以看出,UserExtBuilder并没有从父类User继承来的属性:id、name的填充方法。

解决方案:

在子类和父类中都使用@SuperBuilder,去掉@Builder。但从import的包中也能看到@SuperBuilder注解是experimental实验性的。不知道会不存存在什么潜在风险,慎用。

从反编译后的User 和UserExt类来看:

public abstract static class UserBuilder<C extends User, B extends User.UserBuilder<C, B>>
public abstract static class UserExtBuilder<C extends UserExt, B extends UserExt.UserExtBuilder<C, B>> extends UserBuilder<C, B>

lombok对UserBuilder 与 UserExtBuilder做了继承关系

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • 聊聊Lombok中的@Builder注解使用教程

    目录 Lombok中的@Builder注解的使用 作用 引入依赖 第二步给实体类加上@Builder注解 第三步使用测试使用@Builder注解生成对象 实体类加上@Builder注解之后的编译结果 Lombok中的@Builder注解的使用 作用 @Builder注解的作用主要是用来生成对象,并且可以为对象链式赋值. 引入依赖 因为@Builder注解是lombok中的东西,所以第一步我们需要引入lombok的依赖,如下图: 第二步给实体类加上@Builder注解 第二步我们需要给我们的实体类

  • 使用Lombok @Builder注解导致默认值无效的问题

    目录 @Builder注解导致默认值无效 原因分析 lombok@Builder忽略属性默认值的坑点 1. 简单使用 2. 默认值问题 3. 修改属性值 @Builder注解导致默认值无效 使用Lombok注解可以极高的简化代码量,比较好用的注解除了@Data之外,还有@Builder这个注解,它可以让你很方便的使用builder模式构建对象,但是今天发现@Builder注解会把对象的默认值清掉. 像下面这段代码,会导致对象的name属性变为null: public class BuilderT

  • Java中lombok的@Builder注解的解析与简单使用详解

    Lombok中@Builder用法 1.建造者模式简介:Builder 使用创建者模式又叫建造者模式.简单来说,就是一步步创建一个对象,它对用户屏蔽了里面构建的细节,但却可以精细地控制对象的构造过程. 2.注解类Builder.java注释: * The builder annotation creates a so-called 'builder' aspect to the class that is annotated or the class  * that contains a mem

  • lombok 子类中如何使用@Builder问题

    目录 lombok子类中如何使用@Builder 子类使用lombok的@Builder注解正确姿势 分析一下 lombok子类中如何使用@Builder lombok大家都知道,在使用POJO过程中,它给我们带来了很多便利,省下大量写get.set方法.构造器.equal.toString方法的时间.除此之外,通过@Builder注解,lombok还可以方便的时间建造者模式. 但是,在使用@Builder过程中,我发现了一问题:子类的Builder对象没有父类的属性.这在使用上造成了一定的问题

  • 使用Lombok的@Builder注解带来的两大坑

    目录 一.@Data和@Builder导致无参构造丢失 二.@Builder注解导致默认值无效 三.分析原因 四.总结 一.@Data和@Builder导致无参构造丢失 单独使用@Data注解,是会生成无参数构造方法. 单独使用@Builder注解,发现生成了全属性的构造方法. @Data和@Builder一起用:我们发现没有了默认的构造方法.如果手动添加无参数构造方法或者用@NoArgsConstructor注解都会报错! 两种解决方法 1.构造方法加上@Tolerate 注解,让lombok

  • Lombok同时使⽤@Data和@Builder踩坑总结

    目录 问题背景 Lombok @Data和@Builder分别单独分析用法 解决方法 Lombok原理 总结 问题背景 Lombok使⽤ 同时使⽤@Data和@Builder ,构建无参构造器报错!编译不通过.如下图: Lombok @Data和@Builder分别单独分析用法 Lombok使⽤@Data可以⽣成⽆参构造和类⾥⾯所有属性的getter/setter⽅法.可以简化我们代码的开发.(需要安装Lombok插件和引⼊Lombok依赖). 例如下⾯的⼀个实体类,引⼊Lombok后,可以⾃动

  • 解决Lombok使用@Builder无法build父类属性的问题

    目录 Lombok使用@Builder无法build父类属性 问题描述 解决方案 使用示例 lombok @Builder注解和build父类属性问题 1.简介 2.使用 3.@Builder注解对类做了什么? 4.优缺点 5.问题:@Builder注解不能 build 父类属性 Lombok使用@Builder无法build父类属性 问题描述 实体类使用Lombok的@Builder来实现Builder模式,但是如果使用了extend继承,则子类无法通过Builder来Build父类属性值 解

  • 在Struts2中如何将父类属性序列化为JSON格式的解决方法

    项目前端由于采用Extjs4,列表分页需要返回三个参数:totalCount(记录总数).start(开始位置).limit(每页条数).由于项目中将有很多grid需要分页,因此想将分页信息放到单独的Action类里边,需要分页的Action直接继承该类即可.原struts.xml配置如下: 复制代码 代码如下: <package name="admin" namespace="/admin" extends="json-default"&

  • 用jQuery解决IE不支持的option disable属性

    在IE浏览器中,给select的options设置 disable属性 是没用的. 必须采用一定手段才可以解决这个问题.当然原理就是记住上次选中的 option. this.selectedIndex 就派上用场了. 需要刷新下才可以看到效果,因为输出的都是文本,不存在js加载的情况. 使用jQuery解决IE不支持的option disable属性 $(document).ready(function(){ $(".myselect option:disabled").css('co

  • 解决vue动态为数据添加新属性遇到的问题

    vue为数据添加属性时遇到的坑,通过self.book[i].['cur']=false;动态为数据添加属性时,数据变化了,但是视图没有发生更新. 具体原因不明白.... 解决方法:通过set来添加属性this.set来添加属性this.set(self.book[i],'cur',false); 这样子来设置,就没问题了 以上这篇解决vue动态为数据添加新属性遇到的问题就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们.

  • 解决vue项目使用font-awesome,build后路径的问题

    问题: 项目在本地run情况下显示正常,在build后font-awesome的css文件依赖的几个图标文件引用路径报错. 在webpack.base.conf.js中关于几个文件的配置如下: { test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/, loader: 'url-loader', options: { limit: 10000, name: utils.assetsPath('fonts/[name].[hash:7].[ext]') } } 解决: 通过

  • 解决Vue 给mapState中定义的属性赋值报错的问题

    1. 实践环境 Vue 2.9.6 2. 问题描述 <script> import { mapState } from 'vuex'; export default { name: "displayCount", computed: { ...mapState({ ...略 count: state => state.a.count }) }, methods: { increaseCount () { this.count = this.count + 1 } }

  • 解决vue项目,npm run build后,报路径错的问题

    在build目录下的webpack.prod.conf.js里面: output: { path: config.build.assetsRoot, publicPath: "/dist/", // 添加这行代码,可解决该问题或者publicPath: "./",也可解决 filename: utils.assetsPath('js/[name].[chunkhash].js'), chunkFilename: utils.assetsPath('js/[id].[

  • MyBatis使用级联操作解决lombok构造方法识别失败问题

    先解决一下idea无法识别lombok构造方法的问题,解决方案是在idea的插件中下载并安装lombok插件. MyBatis级联操作,列举最简单的student-classes(学生与班级)的关系表: create table if not exists student ( id int primary key auto_increment, name varchar(20) not null comment '学生姓名', cid int not null comment '班级id' );

随机推荐