您好,登錄后才能下訂單哦!
目標
在SpringBoot接口中,我們一般用@RequestBody類注解需要反序列化的對象,但是當存在多個子類的情況下,常規的反序列化不能滿足需求,比如:
我們有一個類Exam用于表示一張試卷:
@Data public class Exam { private String name; private List<Question> questions; }
這里Question比較特殊,Question本身是一個抽象類,提供了一些通用的方法調用,實際子類有單選題、多選題、判斷題多種情況
實現
SprintBoot內置的序列化是使用的Jackson,查閱文檔后發現Jackson提供了@JsonTypeInfo和@JsonSubTypes這兩個注解,搭配使用,可以根據指定的字段值來指定實例化中用到的具體的子類類型
這幾個類的實際代碼如下:
抽象基類Question:
@Data @JsonTypeInfo( use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.EXISTING_PROPERTY, property = "type", visible = true) @JsonSubTypes({ @JsonSubTypes.Type(value = SingleChoiceQuestion.class, name = Question.SINGLE_CHOICE), @JsonSubTypes.Type(value = MultipleChoiceQuestion.class, name = Question.MULTIPLE_CHOICE), @JsonSubTypes.Type(value = TrueOrFalseQuestion.class, name = Question.TRUE_OR_FALSE), }) public abstract class Question { protected static final String SINGLE_CHOICE = "single_choice"; protected static final String MULTIPLE_CHOICE = "multiple_choice"; protected static final String TRUE_OR_FALSE = "true_or_false"; protected String type; protected String content; protected String answer; protected boolean isCorrect(String answer) { return this.answer.equals(answer); } }
判斷題TrueOrFalseQuestion:
@Data @EqualsAndHashCode(callSuper = true) public class TrueOrFalseQuestion extends Question { public TrueOrFalseQuestion() { this.type = TRUE_OR_FALSE; } }
選擇題ChoiceQuestion:
@Data @EqualsAndHashCode(callSuper = true) public abstract class ChoiceQuestion extends Question { private List<Option> options; @Data public static class Option { private String code; private String content; } }
單選題SingleChoiceQuestion:
@Data @EqualsAndHashCode(callSuper = true) public class SingleChoiceQuestion extends ChoiceQuestion { public SingleChoiceQuestion() { this.type = SINGLE_CHOICE; } }
多選題MultipleChoiceQuestion:
@Data @EqualsAndHashCode(callSuper = true) public class MultipleChoiceQuestion extends ChoiceQuestion { public MultipleChoiceQuestion() { this.type = MULTIPLE_CHOICE; } @Override public void setAnswer(String answer) { this.answer = sortString(answer); } @Override public boolean isCorrect(String answer) { return this.answer.equals(sortString(answer)); } private String sortString(String str) { char[] chars = str.toCharArray(); Arrays.sort(chars); return String.valueOf(chars); } }
測試
接下來測試一下
定義一個接口,我們可以使用@RequestBody傳入一個Exam對象,返回解析結果:
@RequestMapping(value = "/exam", method = RequestMethod.POST) public List<String> parseExam(@RequestBody Exam exam) { List<String> results = new ArrayList<>(); results.add(String.format("Parsed an exam, name = %s", exam.getName())); results.add(String.format("Exam has %s questions", exam.getQuestions().size())) List<String> types = new ArrayList<>(); for (Question question : exam.getQuestions()) { types.add(question.getType()); } results.add(String.format("Questions types: %s", types.toString())); return results; }
項目跑起來,調用接口測試一下:
curl -X POST \ http://127.0.0.1:8080/exam/ \ -H 'Content-Type: application/json' \ -d '{ "name":"一場考試", "questions": [ { "type": "single_choice", "content": "單選題", "options": [ { "code":"A", "content": "選項A" },{ "code":"B", "content": "選項B" }], "answer": "A" },{ "type": "multiple_choice", "content": "多選題", "options": [ { "code":"A", "content": "選項A" },{ "code":"B", "content": "選項B" }], "answer": "AB" },{ "type": "true_or_false", "content": "判斷題", "answer": "True" }] }'
接口返回如下:
[ "Parsed an exam, name = 一場考試", "Exam has 3 questions", "Questions types: [single_choice, multiple_choice, true_or_false]" ]
這里不同類型的question,type字段都能正確讀取,表明反序列化過程中確實是調用了具體子類對應的類來進行實例化的。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對億速云的支持。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。