文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

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

2024-04-02 19:55

关注

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;

@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;

@Setter
@Getter
@NoArgsConstructor
public class Child extends Father {
    
    private String childName;
    
    @Builder(toBuilder = true)
    public Child(String fatherName, String childName) {
        super(fatherName);
        this.childName = childName;
    }
}

使用:

package com.baijia.uqun.individual.management.facade;

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.*;

@Getter
@Setter
@ToString
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class User {
    private Long id;
    private String name;
}

2)创建子类UserExt

import lombok.*;

@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做了继承关系

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

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯