Інший варіант полягає у використанні InjectableValues і @JacksonInject . Це дуже корисно, якщо вам потрібно використовувати не завжди одне і те ж значення, але одне отримати з БД або десь ще для конкретного випадку. Ось приклад використання JacksonInject
:
protected static class Some {
private final String field1;
private final String field2;
public Some(@JsonProperty("field1") final String field1,
@JsonProperty("field2") @JacksonInject(value = "defaultValueForField2",
useInput = OptBoolean.TRUE) final String field2) {
this.field1 = requireNonNull(field1);
this.field2 = requireNonNull(field2);
}
public String getField1() {
return field1;
}
public String getField2() {
return field2;
}
}
@Test
public void testReadValueInjectables() throws JsonParseException, JsonMappingException, IOException {
final ObjectMapper mapper = new ObjectMapper();
final InjectableValues injectableValues =
new InjectableValues.Std().addValue("defaultValueForField2", "somedefaultValue");
mapper.setInjectableValues(injectableValues);
final Some actualValueMissing = mapper.readValue("{\"field1\": \"field1value\"}", Some.class);
assertEquals(actualValueMissing.getField1(), "field1value");
assertEquals(actualValueMissing.getField2(), "somedefaultValue");
final Some actualValuePresent =
mapper.readValue("{\"field1\": \"field1value\", \"field2\": \"field2value\"}", Some.class);
assertEquals(actualValuePresent.getField1(), "field1value");
assertEquals(actualValuePresent.getField2(), "field2value");
}
Майте на увазі, що якщо ви використовуєте конструктор для створення сутності (це зазвичай трапляється, коли ви використовуєте @Value або @AllArgsConstructor в lombok ), і ви ставите @JacksonInject
не до конструктора, а до властивості, яка не буде працювати належним чином - значення введеного поле завжди буде перевизначати значення в JSON, незалежно від того , чи ставити useInput = OptBoolean.TRUE
в @JacksonInject
. Це тому, що jackson вводить ці властивості після виклику конструктора (навіть якщо властивість є final
) - для поля встановлено правильне значення в конструкторі, але тоді воно замінено (перевірте: https://github.com/FasterXML/jackson-databind/ issues / 2678 та https://github.com/rzwitserloot/lombok/issues/1528#issuecomment-607725333для отримання додаткової інформації), цей тест, на жаль, проходить :
protected static class Some {
private final String field1;
@JacksonInject(value = "defaultValueForField2", useInput = OptBoolean.TRUE)
private final String field2;
public Some(@JsonProperty("field1") final String field1,
@JsonProperty("field2") @JacksonInject(value = "defaultValueForField2",
useInput = OptBoolean.TRUE) final String field2) {
this.field1 = requireNonNull(field1);
this.field2 = requireNonNull(field2);
}
public String getField1() {
return field1;
}
public String getField2() {
return field2;
}
}
@Test
public void testReadValueInjectablesIncorrectBehavior() throws JsonParseException, JsonMappingException, IOException {
final ObjectMapper mapper = new ObjectMapper();
final InjectableValues injectableValues =
new InjectableValues.Std().addValue("defaultValueForField2", "somedefaultValue");
mapper.setInjectableValues(injectableValues);
final Some actualValueMissing = mapper.readValue("{\"field1\": \"field1value\"}", Some.class);
assertEquals(actualValueMissing.getField1(), "field1value");
assertEquals(actualValueMissing.getField2(), "somedefaultValue");
final Some actualValuePresent =
mapper.readValue("{\"field1\": \"field1value\", \"field2\": \"field2value\"}", Some.class);
assertEquals(actualValuePresent.getField1(), "field1value");
assertEquals(actualValuePresent.getField2(), "somedefaultValue");
}
Сподіваюся, це допоможе комусь із подібною проблемою.
PS Я використовую jackson v. 2.9.6