Как построить объект, когда у нас есть отношения наследования, используя построитель ломбок?

В моем проекте я использую ломбок, чтобы не писать геттеры и сеттеры для класса. Кроме того, я использую lombok.Builder для создания объекта вместо написания new Obeject() и последующей установки всех значений.

Но когда у нас есть отношения наследования и когда мы хотим создать дочерний объект с помощью построителя ломбоков, я не получаю родительское поле.

Например:

@Data
@NoArgsConstructor
@AllArgsConstructor
@ToString
@EqualsAndHashCode
public class Parent{
  private String nationality;
  .
  .
  // more columns
}

И дочерний класс будет примерно таким:

@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = true)
public class Child extends Parent{
   private String firstName;
   private String lastName;
   .
   .
}

В моем тестовом классе, где мне нужно создать дочерний объект

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();
 }


}

Пожалуйста, дайте мне знать, есть ли способ справиться с этим сценарием на ломбоке.


person Rakesh    schedule 07.10.2015    source источник
comment
Не уверен, почему за него проголосовали. Пожалуйста, дайте мне знать, если я могу что-то улучшить.   -  person Rakesh    schedule 07.10.2015


Ответы (2)


@Builder не может определить, какие поля Parent вы хотите открыть.

Когда @Builder помещается в класс, в *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());
    }
}
person Jeff    schedule 06.11.2015

Приведенное выше решение работает, однако требует слишком много обходного пути. И далее любые изменения в дочернем и родительском классе требуют везде менять аргументы конструктора.

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, как показано ниже (это было невозможно с аннотацией @Builder)

ChildClass.builder().a("testA").b("testB").c("testC").build();
person Amit Kaneria    schedule 07.08.2018