Подтвердить что ты не робот

Lombok builder для проверки не null, а не пустого

У меня есть класс с переменными, я не хочу, чтобы он был пустым или пустым. Есть ли способ использовать Lombok builder для установки свойства? Я могу использовать @NonNull но я не смогу проверить, пуст он или нет. Очевидно, что другой вариант - написать мой собственный строитель, который выполняет все эти проверки. Например:

class Person {
    @NonNull
    private String firstName;
    @NonNull
    private String lastName;

    public static class PersonBuilder() {
        // .
        // .
        // .
        public Person build() {
            //do checks for empty etc and return object
        }
    }
}
4b9b3361

Ответ 1

Максим Кирилов отвечает неполным. Он не проверяет пустые/пустые строки.

Я столкнулся с такой же проблемой раньше, и я понял, что помимо использования @NonNull и @Builder из Lombok перегрузите конструктор с помощью частного модификатора доступа, где вы можете выполнить проверки. Что-то вроде этого:

private Person(final String firstName, final String lastName) {
    if(StringUtils.isBlank(firstName)) {
        throw new IllegalArgumentException("First name can't be blank/empty/null"); 
    }
    if(StringUtils.isBlank(lastName)) {
        throw new IllegalArgumentException("Last name can't be blank/empty/null"); 
    }
    this.firstName = firstName;
    this.lastName = lastName;
}

Кроме того, бросание IllegalArgumentException имеет больше смысла (вместо NPE), когда String имеет пустые, пустые или нулевые значения.

Ответ 2

Аннотации строителя должны решить вашу проблему:

@Builder
class Person {
    @NonNull
    private String firstName;
    @NonNull
    private String lastName;
}

Сгенерированный код:

class Person {
    @NonNull
    private String firstName;
    @NonNull
    private String lastName;

    @ConstructorProperties({"firstName", "lastName"})
    Person(@NonNull String firstName, @NonNull String lastName) {
        if(firstName == null) {
            throw new NullPointerException("firstName");
        } else if(lastName == null) {
            throw new NullPointerException("lastName");
        } else {
            this.firstName = firstName;
            this.lastName = lastName;
        }
    }

    public static Person.PersonBuilder builder() {
        return new Person.PersonBuilder();
    }

    public static class PersonBuilder {
        private String firstName;
        private String lastName;

        PersonBuilder() {
        }

        public Person.PersonBuilder firstName(String firstName) {
            this.firstName = firstName;
            return this;
        }

        public Person.PersonBuilder lastName(String lastName) {
            this.lastName = lastName;
            return this;
        }

        public Person build() {
            return new Person(this.firstName, this.lastName);
        }

        public String toString() {
            return "Person.PersonBuilder(firstName=" + this.firstName + ", lastName=" + this.lastName + ")";
        }
    }
}

В этом случае нулевая валидация будет иметь место при построении объекта.

Ответ 3

Я сделал что-то вроде этого,

class Person {
    private String mFristName;
    private String mSecondName;

    @Builder
    Person(String firstName, String secondName) {
        mFristName = PreCondition.checkNotNullOrEmpty(firstName);
        mSecondName = PreCondition.checkNotNullOrEmpty(secondName);
    }
}

class PreCondition {

    static <T> T checkNotNullOrEmpty(T instance) {
        if (instance == null || (instance instanceof String && ((String) instance).isEmpty())) {
            throw new NullOrEmptyException();
        }
        return instance;
    }

    static class NullOrEmptyException extends RuntimeException {
        NullOrEmptyException() {
            super("Null or Empty");
        }
    }
}