Come deserializzare la stringa JSON da enumerare
Ho il JSON simile al seguente:
{
"name": "john",
"options": {
"test": 1,
"operation": "op1", // I need to deserialize this option to enum.
...
// any number of options
}
}
e ho la classe che assomiglia al seguente:
public class Info {
public String name;
@JsonTypeInfo(use = JsonTypeInfo.Id.NONE)
public Map<String, Object> options;
}
public enum OperationEnum {OP1, OP2, OP3}
Come posso deserializzare per optionsmappare operationlike enume testtoInteger
Risposte
Dovresti essere in grado di utilizzare @JsonAnySetterper aiutarti qui:
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}
C'è un buon articolo qui: https://www.concretepage.com/jackson-api/jackson-jsonanygetter-and-jsonanysetter-example
Perché non farlo in questo modo:
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}
Allora dovrebbe JustWork!
La struttura JSON, in cui tutte le opzioni (di diversi tipi) sono memorizzate come Map<String, Object> options;impone un'analisi extra, perché il tipo di valore viene identificato in base all'etichetta (come "operation"e OperationEnumclasse) Qui ci sono 2 approcci con ObjectMapper.readValueeEnum.valueOf
Se possibile, considera soluzioni più flessibili come quelle di @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" +
"}";
}
La soluzione semplice è racchiudere l' Mapistanza con una classe e implementare metodi aggiuntivi per ogni proprietà non regolare. Puoi utilizzare l' @JsonAnySetterannotazione per memorizzare tutte le key-valuecoppie.
Vedi esempio sotto:
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}
Stampe sopra il codice:
{test=1, operation=op1}
OP1