使用@Builder导致无法创建无参构造方法的解决

 更新时间:2023年12月15日 15:10:51   作者:搏·梦  
这篇文章主要介绍了使用@Builder导致无法创建无参构造方法的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

1. 前言

在创建实体类的时候,有时候参数过于多,如果使用有参数的构造函数,实在看起来不是很优雅,一般都会想使用@Builder构建者模式来进行创建对象,不仅仅优雅,而且还方便。

当偶然机会发现,如果使用了@Builder,该实体类会失去无参构造方法

具体现象如下:

当不加任何注解的时候,完成正常:

当加了@Builder的时候,则会直接出现下面的错误,提示没有无参的构造函数:

2. 先说结论

由于没有时间深入查看@Builder的源码,但是可以从编译之后的字节码得出结论: 实体类使用了@Builder,会有全参构造函数,没有默认无参构造函数,即无法使用默认无参构造函数。

解决方法:

  • 在实体类上同时加上:@Builder、@AllArgsConstructor、@NoArgsConstructor
  • 手写全部有参构造方法,以及 无参构造方法
  • 手写个默认无参构造方法,并加上@Tolerate
@Builder
public class demoBuilderEntry {
    private String name;
    private String sex;

    @Tolerate
    public demoBuilderEntry() {
    }
}

3. 演示

若不知道怎么查看java反编译字节码内容,可以看一下:

Java如何进行反编译生成.java文件(javap、jad下载安装使用)

1.当实体类不加任何东西:

public class demoBuilderEntry {
    private String name;
    private String sex;
}

可以看见是默认有无参构造函数

public class demoBuilderEntry {

    public demoBuilderEntry() {
    }

    private String name;
    private String sex;
}

2.当实体类加上@Builder注解:

@Builder
public class demoBuilderEntry {
    private String name;
    private String sex;
}

可以看出:无参构造函数已不存在,只剩全参构造函数

public class demoBuilderEntry {
    public static class demoBuilderEntryBuilder {

        public demoBuilderEntryBuilder name(String name) {
            this.name = name;
            return this;
        }

        public demoBuilderEntryBuilder sex(String sex) {
            this.sex = sex;
            return this;
        }

        public demoBuilderEntry build() {
            return new demoBuilderEntry(name, sex);
        }

        public String toString() {
            return (new StringBuilder()).append("demoBuilderEntry.demoBuilderEntryBuilder(name=").append(name).append(", sex=").append(sex).append(")").toString();
        }

        private String name;
        private String sex;

        demoBuilderEntryBuilder() {
        }
    }

    demoBuilderEntry(String name, String sex) {
        this.name = name;
        this.sex = sex;
    }

    public static demoBuilderEntryBuilder builder() {
        return new demoBuilderEntryBuilder();
    }

    private String name;
    private String sex;
}

因此,如下图,一定会报错:

有人会说:没有无参怎么了,也没有什么事情,反正需要构造。

但像spring这种,我们可以从配置文件中注入属性,如@ConfigurationProperties,它们的做法是先创建该类对象,使用无参构造函数,然后再调用set方法,此时,你没有默认无参构造方法,必然会失败,毕竟连对象都没有创建成功。

具体失败案例可以看:实体类使用@Builder,导致@ConfigurationProperties注入属性失败

3.如果需要无参构造函数,那就手动添加,在实体类上,再加上@NoArgsConstructor也无效。

根据下图可以看见,似乎没有报错

当运行的时候,则报错如下:与实际参数列表不符,显然是无法通过加@NoArgsConstructor创建默认无参构造函数

4. 解决方法

1. 在实体类上再加上@AllArgsConstructor即可

根据下图查看,似乎也没有问题:

点击运行,查看结果的时候,居然可以运行成功如下:

反手我们去看一下反编译:

public class demoBuilderEntry {
    public static class demoBuilderEntryBuilder {
        // builder 的代码省略,没有粘出来
    }
    
    public static void main(String args[]) {
        System.out.println(new demoBuilderEntry());
    }

    public static demoBuilderEntryBuilder builder() {
        return new demoBuilderEntryBuilder();
    }

    public demoBuilderEntry(String name, String sex) {
        this.name = name;
        this.sex = sex;
    }
	// 无参构造函数有了
    public demoBuilderEntry() {
    }

    private String name;
    private String sex;
}

2. 手写个无参构造函数并加上@Tolerate

@Tolerate:中文:容许,通俗来说,被该注解加上的方法,让lombok容许它们存在。

/**
 * Put on any method or constructor to make lombok pretend it doesn't exist,
 * i.e., to generate a method which would otherwise be skipped due to possible conflicts.
 */
@Target({ElementType.METHOD, ElementType.CONSTRUCTOR})
@Retention(RetentionPolicy.SOURCE)
public @interface Tolerate {
}

测试:

3. 手写有参无参构造方法

这种方法是最简单粗暴的,但是代码可能会没有那么简洁:

总结

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

相关文章

  • servlet实现简单的权限管理和敏感词过滤功能

    servlet实现简单的权限管理和敏感词过滤功能

    JavaEE课要求用servlet和过滤器实现权限管理和敏感词过滤功能,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-05-05
  • Java 泛型(Generic)简介及用法详解

    Java 泛型(Generic)简介及用法详解

    泛型是一种把类型明确的工作推迟到创建对象或者调用方法的时候才去明确的特殊的类型,参数化类型,把类型当作参数一样的传递,本文给大家介绍Java 泛型(Generic)概述及使用,感兴趣的朋友跟随小编一起看看吧
    2023-10-10
  • Java并发编程中的阻塞队列解析

    Java并发编程中的阻塞队列解析

    这篇文章主要介绍了Java并发编程中的阻塞队列解析,阻塞队列BlockingQueue是一个支持两个附加操作的队列,这两个附加的操作是在队列为空时,获取元素的线程会等待队列变为非空,当队列满时,存储元素的线程会等待队列可用,需要的朋友可以参考下
    2023-08-08
  • Java中实现将jar包内文件资源释放出来

    Java中实现将jar包内文件资源释放出来

    这篇文章主要介绍了Java中实现将jar包内文件资源释放出来的方法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-08-08
  • Java中使用同步回调和异步回调的示例详解

    Java中使用同步回调和异步回调的示例详解

    这篇文章主要介绍了Java中使用同步回调和异步回调的相关资料,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-04-04
  • Java Socket实现聊天室附1500行源代码

    Java Socket实现聊天室附1500行源代码

    Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。本篇文章手把手带你通过Java Socket来实现自己的聊天室,大家可以在过程中查缺补漏,温故而知新
    2021-10-10
  • Java的Socket实现长连接以及数据的发送和接收方式

    Java的Socket实现长连接以及数据的发送和接收方式

    这篇文章主要介绍了Java的Socket实现长连接以及数据的发送和接收方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-09-09
  • MybatisPlus代码生成器含XML文件详解

    MybatisPlus代码生成器含XML文件详解

    这篇文章主要介绍了MybatisPlus代码生成器含XML文件详解,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-01-01
  • 如何处理器拦截器(HandlerInterceptor)

    如何处理器拦截器(HandlerInterceptor)

    这篇文章主要介绍了如何处理器拦截器(HandlerInterceptor)问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • java 实现定时的方法及实例代码

    java 实现定时的方法及实例代码

    这篇文章主要介绍了java 定时任务详细介绍及实例代码的相关资料,需要的朋友可以参考下
    2017-03-03

最新评论