Jak deserializować ciąg JSON do wyliczenia
Mam JSON wygląda następująco:
{
"name": "john",
"options": {
"test": 1,
"operation": "op1", // I need to deserialize this option to enum.
...
// any number of options
}
}
a u mnie klasa wygląda następująco:
public class Info {
public String name;
@JsonTypeInfo(use = JsonTypeInfo.Id.NONE)
public Map<String, Object> options;
}
public enum OperationEnum {OP1, OP2, OP3}
Jak mogę deserializować, aby options
mapować operation
jak enum
i test
doInteger
Odpowiedzi
Powinieneś być w stanie skorzystać @JsonAnySetter
z pomocy tutaj:
public class Info {
public static class Options {
public int test;
public OperationEnum operation;
}
public String name;
public Options options;
private final Map<String, Object> properties = new HashMap<>();
@JsonAnySetter
public void add(String key, String value) {
properties.put(key, value);
}
}
public enum OperationEnum {OP1, OP2, OP3}
Tutaj jest dobry napis: https://www.concretepage.com/jackson-api/jackson-jsonanygetter-and-jsonanysetter-example
Dlaczego nie zrobić tego w ten sposób:
public class Info {
public static class Options {
public int test;
public OperationEnum operation;
}
public String name;
public Options options;
}
public enum OperationEnum {OP1, OP2, OP3}
Wtedy powinno JustWork!
Struktura JSON, w której przechowywane są wszystkie opcje (różnego rodzaju), tak jak Map<String, Object> options;
wymusza dodatkowe analizowanie, ponieważ typ wartości jest identyfikowany na podstawie etykiety (jak "operation"
i OperationEnum
klasa) Oto 2 podejścia z ObjectMapper.readValue
iEnum.valueOf
Jeśli to możliwe, rozważ bardziej elastyczne rozwiązania, takie jak te od @Matthew
class Demo {
public static void main(String[] args) throws Exception {
ObjectMapper om = new ObjectMapper();
Info info = om.readValue(jsonValue, Info.class);
OperationEnum operationM = om.readValue("\"" + info.options.get("operation") + "\"", OperationEnum.class);
System.out.println(operationM);
OperationEnum operationE = OperationEnum.valueOf(info.options.get("operation").toString());
System.out.println(operationE);
}
static String jsonValue = "{\n" +
" \"name\": \"john\",\n" +
" \"options\": {\n" +
" \"test\": 1,\n" +
" \"operation\": \"OP1\" \n" +
" }\n" +
"}";
}
Prostym rozwiązaniem jest otoczenie Map
instancji klasą i zaimplementowanie dodatkowych metod dla każdej niestandardowej właściwości. Możesz użyć @JsonAnySetter
adnotacji do przechowywania wszystkich key-value
par.
Zobacz poniższy przykład:
import com.fasterxml.jackson.annotation.JsonAnySetter;
import com.fasterxml.jackson.databind.ObjectMapper;
import java.io.File;
import java.io.IOException;
import java.util.Arrays;
import java.util.LinkedHashMap;
import java.util.Map;
public class JsonOptionsWithCustomEnumApp {
public static void main(String[] args) throws IOException {
File jsonFile = new File("./resource/test.json").getAbsoluteFile();
ObjectMapper mapper = new ObjectMapper();
Info info = mapper.readValue(jsonFile, Info.class);
Options options = info.getOptions();
System.out.println(options.toMap());
System.out.println(options.getOperation());
}
}
class Info {
private String name;
private Options options;
//getters, setters
}
class Options {
private Map<String, Object> values = new LinkedHashMap<>();
@JsonAnySetter
private void setValues(String key, Object value) {
values.put(key, value);
}
public OperationEnum getOperation() {
Object operation = values.get("operation");
if (operation == null) {
return null;
}
return Arrays.stream(OperationEnum.values())
.filter(op -> op.name().equalsIgnoreCase(operation.toString()))
.findFirst()
.orElse(null);
}
public Map<String, Object> toMap() {
return values;
}
}
enum OperationEnum {OP1, OP2, OP3}
Powyższy kod drukuje:
{test=1, operation=op1}
OP1