当我们使用 lombok 构建器有继承关系时,如何构建一个 object?
How to build an object when we have inheritance relationship using lombok builder?
在我的项目中,我使用 lombok 来避免为 class 编写 getter 和 setter。
此外,我正在使用 lombok.Builder 构建一个 object 而不是编写 new Obeject() 然后设置所有值。
但是当我们有继承关系时,当我们想使用lombok builder构造child object时,我没有得到parent的字段。
例如:
@Data
@NoArgsConstructor
@AllArgsConstructor
@ToString
@EqualsAndHashCode
public class Parent{
private String nationality;
.
.
// more columns
}
和Child class会是这样的:
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = true)
public class Child extends Parent{
private String firstName;
private String lastName;
.
.
}
在我的测试中 class,我需要在其中构建 child object
public class Test{
public void testMethod(){
Child child = Child.builder()
.firstName("Rakesh")
.lastName("SS")
.nationality("some text")// I am not able to set nationality
.build();
}
}
请告诉我,在 lombok 中有什么方法可以处理这种情况。
@Builder
无法确定您希望公开 Parent
的哪些字段。
当 @Builder
放在 class 上时,只有在 class 上明确声明的字段才会添加到 *Builder
。
当 @Builder
放在静态方法或构造函数上时,生成的 *Builder
将为每个参数提供一个方法。
此外,如果您使用的是 @Builder
,那么可以安全地假设至少 Child
是不可变的吗?
我提供了两个例子,一个是 Parent
是可变的,Child
是不可变的,另一个是 Parent
和 Child
都是不可变的。
不可变的父子
import static org.junit.Assert.*;
import lombok.Builder;
import lombok.EqualsAndHashCode;
import lombok.ToString;
import lombok.Value;
import lombok.experimental.NonFinal;
import org.junit.Test;
public class So32989562ValueTest {
@Value
@NonFinal
public static class Parent {
protected final String nationality;
}
@Value
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = true)
public static class Child extends Parent {
private final String firstName;
private final String lastName;
@Builder(toBuilder = true)
private Child(String nationality, String firstName, String lastName) {
super(nationality);
this.firstName = firstName;
this.lastName = lastName;
}
}
@Test
public void testChildBuilder() {
String expectedFirstName = "Jeff";
String expectedLastName = "Maxwell";
String expectedNationality = "USA";
Child result = Child.builder()
.firstName(expectedFirstName)
.lastName(expectedLastName)
.nationality(expectedNationality)
.build();
assertEquals(result.toString(), expectedFirstName, result.getFirstName());
assertEquals(result.toString(), expectedLastName, result.getLastName());
assertEquals(result.toString(), expectedNationality, result.getNationality());
}
}
可变父项,不可变子项:
import static org.junit.Assert.*;
import lombok.Builder;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.ToString;
import lombok.Value;
import org.junit.Test;
public class So32989562DataTest {
@Data
public static class Parent {
protected String nationality;
}
@Value
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = true)
public static class Child extends Parent {
private final String firstName;
private final String lastName;
@Builder(toBuilder = true)
private Child(String nationality, String firstName, String lastName) {
this.setNationality(nationality);
this.firstName = firstName;
this.lastName = lastName;
}
}
@Test
public void testChildBuilder() {
String expectedFirstName = "Jeff";
String expectedLastName = "Maxwell";
String expectedNationality = "USA";
Child result = Child.builder()
.firstName(expectedFirstName)
.lastName(expectedLastName)
.nationality(expectedNationality)
.build();
assertEquals(result.toString(), expectedFirstName, result.getFirstName());
assertEquals(result.toString(), expectedLastName, result.getLastName());
assertEquals(result.toString(), expectedNationality, result.getNationality());
}
}
以上解决方案有效,但这需要太多的解决方法。此外,子项和父项的任何更改 class 都需要在所有地方更改构造函数参数。
Lombok 在版本 1.18.2 中引入了实验性功能,用于解决 Builder 注解面临的继承问题,可以通过 @SuperBuilder 注解解决,如下所示。
@SuperBuilder
public class ParentClass {
private final String a;
private final String b;
}
@SuperBuilder
public class ChildClass extends ParentClass{
private final String c;
}
现在,可以如下使用 Builder class(@Builder 注释无法实现)
ChildClass.builder().a("testA").b("testB").c("testC").build();
在我的项目中,我使用 lombok 来避免为 class 编写 getter 和 setter。 此外,我正在使用 lombok.Builder 构建一个 object 而不是编写 new Obeject() 然后设置所有值。
但是当我们有继承关系时,当我们想使用lombok builder构造child object时,我没有得到parent的字段。
例如:
@Data
@NoArgsConstructor
@AllArgsConstructor
@ToString
@EqualsAndHashCode
public class Parent{
private String nationality;
.
.
// more columns
}
和Child class会是这样的:
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = true)
public class Child extends Parent{
private String firstName;
private String lastName;
.
.
}
在我的测试中 class,我需要在其中构建 child object
public class Test{
public void testMethod(){
Child child = Child.builder()
.firstName("Rakesh")
.lastName("SS")
.nationality("some text")// I am not able to set nationality
.build();
}
}
请告诉我,在 lombok 中有什么方法可以处理这种情况。
@Builder
无法确定您希望公开 Parent
的哪些字段。
当 @Builder
放在 class 上时,只有在 class 上明确声明的字段才会添加到 *Builder
。
当 @Builder
放在静态方法或构造函数上时,生成的 *Builder
将为每个参数提供一个方法。
此外,如果您使用的是 @Builder
,那么可以安全地假设至少 Child
是不可变的吗?
我提供了两个例子,一个是 Parent
是可变的,Child
是不可变的,另一个是 Parent
和 Child
都是不可变的。
不可变的父子
import static org.junit.Assert.*;
import lombok.Builder;
import lombok.EqualsAndHashCode;
import lombok.ToString;
import lombok.Value;
import lombok.experimental.NonFinal;
import org.junit.Test;
public class So32989562ValueTest {
@Value
@NonFinal
public static class Parent {
protected final String nationality;
}
@Value
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = true)
public static class Child extends Parent {
private final String firstName;
private final String lastName;
@Builder(toBuilder = true)
private Child(String nationality, String firstName, String lastName) {
super(nationality);
this.firstName = firstName;
this.lastName = lastName;
}
}
@Test
public void testChildBuilder() {
String expectedFirstName = "Jeff";
String expectedLastName = "Maxwell";
String expectedNationality = "USA";
Child result = Child.builder()
.firstName(expectedFirstName)
.lastName(expectedLastName)
.nationality(expectedNationality)
.build();
assertEquals(result.toString(), expectedFirstName, result.getFirstName());
assertEquals(result.toString(), expectedLastName, result.getLastName());
assertEquals(result.toString(), expectedNationality, result.getNationality());
}
}
可变父项,不可变子项:
import static org.junit.Assert.*;
import lombok.Builder;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.ToString;
import lombok.Value;
import org.junit.Test;
public class So32989562DataTest {
@Data
public static class Parent {
protected String nationality;
}
@Value
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = true)
public static class Child extends Parent {
private final String firstName;
private final String lastName;
@Builder(toBuilder = true)
private Child(String nationality, String firstName, String lastName) {
this.setNationality(nationality);
this.firstName = firstName;
this.lastName = lastName;
}
}
@Test
public void testChildBuilder() {
String expectedFirstName = "Jeff";
String expectedLastName = "Maxwell";
String expectedNationality = "USA";
Child result = Child.builder()
.firstName(expectedFirstName)
.lastName(expectedLastName)
.nationality(expectedNationality)
.build();
assertEquals(result.toString(), expectedFirstName, result.getFirstName());
assertEquals(result.toString(), expectedLastName, result.getLastName());
assertEquals(result.toString(), expectedNationality, result.getNationality());
}
}
以上解决方案有效,但这需要太多的解决方法。此外,子项和父项的任何更改 class 都需要在所有地方更改构造函数参数。
Lombok 在版本 1.18.2 中引入了实验性功能,用于解决 Builder 注解面临的继承问题,可以通过 @SuperBuilder 注解解决,如下所示。
@SuperBuilder
public class ParentClass {
private final String a;
private final String b;
}
@SuperBuilder
public class ChildClass extends ParentClass{
private final String c;
}
现在,可以如下使用 Builder class(@Builder 注释无法实现)
ChildClass.builder().a("testA").b("testB").c("testC").build();