解决lombok 父类和子类builder不兼容的问题
作者:Armour__r
遇到的问题
在写代码时,有时因为需要定义一些重复的参数,为了复用之前传参的DTO,会对原有的类进行继承,从而达到避免重复代码的效果。
但是,当父类中使用了lombok的@Builder注解,子类也需要@builder注解时,就会出现异常
排查和解决
由于实际的代码存在敏感信息,这里分别定义两个类Parent和Child来进行场景的模拟
@Data @Builder class Parent { private String parentProperty1; private String parentProperty2; } @EqualsAndHashCode(callSuper = true) @Data @Builder class Child extends Parent { private String childProperty1; }
这时在尝试使用子类的builder方法时,发现没有办法链式调用,使用时只能初始化子类中的变量,编译时会直接失败,抛出异常
Error:(160, 1) java: 无法将类 org.example.Parent中的构造器 Parent应用到给定类型;
需要: java.lang.String,java.lang.String
找到: 没有参数
原因: 实际参数列表和形式参数列表长度不同
这是在编译子类的@Builder注解时出现的异常,原因直观的看起来是找不到构造器,在Parent类上加上@NoArgsConstructor和@AllArgsConstructor这两个注解就能解决这个问题,但是同时会出现新的编译问题,是什么问题先按下不表。
想要简单的解决加上@Builder之后就会报错的问题,那么直接把父类的@Builder这个注解拿掉就行了,不过这时无法设置父类的属性,如果还想在子类中使用构建器模式来初始化父类的属性,还有另一种方法,在子类中实现一个能够初始化父类属性的构造器,并在这个构造方法上添加@Builder注解。
这时的代码:
@Data @NoArgsConstructor @AllArgsConstructor class Parent { private String parentProperty1; private String parentProperty2; } @Data class Child extends Parent { private String childProperty1; @Builder public Child(String parentProperty1, String parentProperty2, String childProperty1){ super(parentProperty1, parentProperty2); this.childProperty1 = childProperty1; } }
不过使用这种方法只能解决子类使用@Builder的问题,但是在更多的时候,父类也是需要@Builder这个注解的,那么在这种情况下应该怎么解决呢?
而且这时还会有另一个新的问题出现,使用了@Data注解和@Builder注解的子类无法使用无参构造器来创建对象,这时需要在子类上显式的加上@NoArgsConstructor这个注解才能解决。
如果要更细致的分析,就得从从@Builder的原理说起,了解@Builder到底生成了哪些代码?
这一步可以自己编译代码看看,当然如果自己写过builder建造者模式的实现,应该能想到他是实现了一个名称以Builder为后缀的静态内部类,在调用build()方法的时候调用外部类的全参构造方法来生成外部类的实例。
回到之前的问题,当子类和父类同时存在@Builder注解时,在解决了构造器异常之后,如果编译代码,会出现异常:
Error:(164, 5) java: org.example.Child中的builder()无法覆盖org.example.Parent中的builder()
返回类型org.example.Child.ChildBuilder与org.example.Parent.ParentBuilder不兼容
这里的问题就简单一些了,父类的builder()方法返回的是ParentBuilder这个静态内部类类型的对象,而子类生成的builder()方法返回的是ChildBuilder这个类型的对象。
两者的名称重复了,而由于返回类型不兼容而无法按覆盖。
根据@Builder注解的源码可以发现名称是可以自定义的,于是可以通过给子类builder方法自定义名称的方式来解决这个问题。
最终的代码:
@Data @NoArgsConstructor @AllArgsConstructor @Builder class Parent { private String parentProperty1; private String parentProperty2; } @Data class Child extends Parent { private String childProperty1; @Builder(builderMethodName = "childBuilder") public Child(String parentProperty1, String parentProperty2, String childProperty1){ super(parentProperty1, parentProperty2); this.childProperty1 = childProperty1; } }
结尾
值得一提的是,1.8.2之后版本的lombok提供了一个新的注解@SuperBuilder来解决这个问题,不过我没有用过,而且从网上搜索出来的结果来看,还是存在一些问题的,建议谨慎升级。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。