Правильный подход (шаблон проектирования) для обновления полей в Java-объекте - PullRequest
0 голосов
/ 14 июня 2019

У меня есть класс Java Person:

public class Person {
    private String name;
    private int age;
    private String marriedStatus;
    private Date dob;
    //getters and setters
}

Когда я получаю новые значения для некоторых полей этого объекта, я могу обновить его.Но в новых полях значения дохода в этом формате: Map<String, String> newValues, где ключ - номер поля, а значение - значение поля.Я создаю этот сервис:

public class UpdateService {
    public Person updateFields(Person targetPerson, Map<String, String> newValues){
        return null;
    }
}

Я создаю модульный тест и прошу вас помочь в его реализации.

public class UpdateServiceTest {

    /*associations between field number and field name
    12 - name (length min: 2, max: 20. First letter must uppercase )
    18 - marriedStatus (only married, divorced, single)
    21  - age (only between 18 and 120)
    14  - dob (some format)
     */

    private Date dob;

    @Before
    public void setUp() {
        dob = new GregorianCalendar(2000, Calendar.NOVEMBER, 20).getTime();
    }

    @Test
    public void returnPersonWithUpdatedFields() {
        UpdateService updateService = new UpdateService();

        Person targetPerson = new Person();
        targetPerson.setName("Name");
        targetPerson.setMarriedStatus("MarriedStatus");
        targetPerson.setAge(20);
        targetPerson.setDob(dob);

        Map<String, String> newValues = new HashMap<String, String>();
        newValues.put("12", "Bill");
        newValues.put("18", "married ");
        newValues.put("21", "25");

        Person person = updateService.updateFields(targetPerson, newValues);

        assertEquals("Bill", person.getName());
        assertEquals("married", person.getMarriedStatus());
        assertEquals(25, person.getAge());
        assertEquals(dob, person.getDob());
    }
}

Мне нужно вызвать человека и обновить только те поля, доход которых в Map<String, String> newValues.И подтвердите это.

1 Ответ

1 голос
/ 14 июня 2019

Вот предложение, как вы могли бы сделать это.

Моделируйте каждое поле как отдельный класс, реализуя общий интерфейс Field.

Внедрите все известные поля через конструктор и когдадля каждой записи ищите соответствующий класс поля для обработки проверки и обновления.

При таком подходе вы можете проверить логику проверки и обновления каждого поля отдельно.Добавление полей пользователя не увеличит ваш класс обслуживания.Тестирование PersonUpdateService должно будет использовать только одно или два проверенных поля для проверки логики поиска и выполнения.Хорошее разделение проблем, я бы сказал.

import java.util.*;

@Component
public class PersonUpdateService {

    private final List<Field> fields;

    @Autowired
    public PersonUpdateService(final List<Field> fields) {
        this.fields = fields;
    }

    public void updatePerson(final Person person, final Map<String, String> update) {
        final boolean updated = false;
        update.forEach((key, value) -> this.findField(key).update(person, value));
    }

    private Field findField(final String index) {
        return this.fields.stream().filter(f -> f.index().equals(index)).findAny().orElseThrow(
                () -> new IllegalArgumentException("Field not found: " + index));
    }

}

Интерфейс поля:

public interface Field {
    String index();

    void update(Person person, String newValue);
}

Пример реализации поля:

import java.util.regex.Pattern;

@Component
public class NameField implements Field {

    private static final String INDEX = "12";
    private static final String REGEX = "/^[A-Z][a-z0-9_-]{1,19}$/";
    private static final String CONSTRAINTS = "length min: 2, max: 20. First letter must uppercase";

    @Override
    public String index() {
        return INDEX;
    }

    @Override
    public void update(final Person person, final String newValue) {
        if (!Pattern.matches(REGEX, newValue)) {
            throw new ValidationException(CONSTRAINTS);
        }
        person.setName(newValue);
    }

}

РЕДАКТИРОВАТЬ: добавлено @Компонент и аннотации @Autowired, чтобы указать, как внедрение зависимости может использоваться в Spring.Доступные компоненты, реализующие интерфейс Field, будут автоматически собраны Spring и введены через конструктор.В своем модульном тесте для сервиса вы можете ввести одно или два ложных поля.Не проверяйте фактическую логику проверки / обновления реализаций полей в сервисном тесте, но создавайте отдельные модульные тесты для каждого класса полей.

РЕДАКТИРОВАТЬ 2: приведенный выше совет по написанию модульных тестов взят из моего (mockist) перспектива.Классические юнит-тестеры, вероятно, написали бы один тест, чтобы охватить все спецификации (как тест, который вы предоставили в посте).Причина, по которой я не предпочитаю, заключается в том, что пограничные случаи легче теряются в таком модульном тесте в стиле интеграции, и увеличивается вероятность того, что вы делаете ложные предположения о том, как работает код, или что вам приходится многократно использовать один и тот же код в вашемтесты.Однако это давняя дискуссия, и существует множество точек зрения, которые имеют свои достоинства.

...