java - 使用 Java 中的 JSON Schema 和 Jackson Schema 解析器库查找任何 JSON 元素的 TYPE
问题描述
我正在尝试type
使用最节省内存和高性能的方法来查找 JSON Schema 中的任何元素。但是,在达到某个点后,我有点卡住并且没有想法,所以希望在这里得到答案:
基本上,我有一个List: jsonPath
由元素及其父元素组成,我需要从Json Schema
我拥有的元素中识别其类型。为了解析Json Schema
我正在使用Java Jackson Library
. 我能够获得所有元素并能够找到它的类型,但在比较过程中,我有点困惑。
以下是我到目前为止的代码:正如我们所见,我有jsonPath
as"food", "Ingredients", "ingredient"
这意味着我需要找到type
最后一个元素的ingredient
,而其他元素是它的父母。该程序应返回type
为array
. 同样,如果我传递任何其他元素,jsonPath array
例如 justfood
或price
then 它应该检查相应的父母和他们的孩子,那么最终应该返回它的类型。我Stack elements
只是出于参考目的,您可以忽略它。
我可以尝试if
在每个阶段添加多个条件并获取元素类型。但是,我试图找到一种简单且更好的方法,而不是使用复杂的嵌套 if。应该可以以一种非常简单的方式进行操作,但目前对我来说没有任何点击。
public class JsonElementLocator {
private final JsonNode outerNode;
private static JsonElementLocator _instance;
// Default constructor to get the content and store in the root
private JsonElementLocator() throws IOException {
final ObjectMapper mapper = new ObjectMapper();
final JsonNode root = mapper.readTree(JsonElementLocator.class.getClassLoader().getResource("testJSON.json"));
outerNode = root.path("definitions");
}
// Method to create a Object instance of current class
private static synchronized JsonElementLocator getInstance() throws IOException {
if (_instance == null) {
_instance = new JsonElementLocator();
}
return _instance;
}
// Method called by other application to get the element Type
public static Optional < String > locate(List < String > elementPath) throws IOException {
JsonElementLocator cl = getInstance();
JsonNode rootNode = cl.outerNode;
recurse(rootNode, new Stack < String > ());
return Optional.empty();
}
// Method called recursively to get the element Type from JSON Schema
private static Optional < String > recurse(JsonNode rootNode, Stack < String > elements) {
// Loop through each Object in Root Node
for (JsonNode childNode: rootNode) {
// Check if childNode is of Object type
if (childNode.path("type").toString().contains("object")) {
elements.push(childNode.get("title").toString());
// Call recurse method for the subsequent objects
recurse(childNode.path("properties"), elements);
elements.pop();
} else if (childNode.path("type").toString().contains("array")) {
// Check if childNode is of Array type and call its elements
for (JsonNode ccNode: childNode.path("items")) {
final String def = ccNode.toString().contains("definitions") ? ccNode.toString().replace("\"", "").replaceAll("#/definitions/",
"") : null;
// Find the element from the root
if (def != null) {
final JsonNode defInfo = _instance.outerNode.get(def);
// Call recurse method for definition elements
recurse(defInfo.path("properties"), elements);
}
}
} else {
System.out.println(String.join("/", elements));
// Check if childNode is of normal type
}
}
return Optional.empty();
}
public static void main(String[] args) throws IOException {
List < String > jsonPath = new ArrayList < String > (List.of("food", "Ingredients", "ingredient"));
locate(jsonPath);
}
}
以下是Json Schema
我要解析的完整内容:
{
"$schema": "http://json-schema.org/draft-06/schema#",
"$ref": "#/definitions/Welcome2",
"definitions": {
"Welcome2": {
"type": "object",
"additionalProperties": false,
"properties": {
"hotelName": {
"type": "string"
},
"food": {
"type": "array",
"items": {
"$ref": "#/definitions/Food"
}
}
},
"required": [
"food",
"hotelName"
],
"title": "Welcome2"
},
"Food": {
"type": "object",
"additionalProperties": false,
"properties": {
"name": {
"type": "string"
},
"price": {
"type": "string"
},
"description": {
"type": "string"
},
"calories": {
"type": "string",
"format": "integer"
},
"ingredients": {
"$ref": "#/definitions/Ingredients"
}
},
"required": [
"calories",
"description",
"ingredients",
"name",
"price"
],
"title": "Food"
},
"Ingredients": {
"type": "object",
"additionalProperties": false,
"properties": {
"ingredient": {
"type": "array",
"items": {
"type": "string"
}
}
},
"required": [
"ingredient"
],
"title": "Ingredients"
}
}
}
解决方案
在尝试了很多事情之后,我能够做到。添加代码作为答案,因为它可以帮助将来尝试实现类似目标的人:
import java.io.IOException;
import java.util.List;
import java.util.Optional;
import java.util.Stack;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
public class TypeLocator {
private final JsonNode outerNode;
private static TypeLocator _instance;
private final JsonNode root;
// Default constructor to get the content and store in the root
private TypeLocator() throws IOException {
// Create the jackson instance and get the JSON Schema Information
final ObjectMapper mapper = new ObjectMapper();
root = mapper.readTree(TypeLocator.class.getClassLoader().getResource("JSON-Schema.json"));
outerNode = root.path("definitions");
}
// Method to create a Object instance of current class
private static synchronized TypeLocator getInstance() throws IOException {
if (_instance == null) {
_instance = new TypeLocator();
}
return _instance;
}
// Method called by other application to get the element Type
public static Optional<String> locate(List<String> elementPath) throws IOException {
// Get the JSON Schema information
TypeLocator cl = getInstance();
JsonNode rootNode = cl.outerNode;
// Check if there are element and obtain the event type info
final String eventType = elementPath.get(0).substring(0, 1).toLowerCase() + elementPath.get(0).substring(1);
final JsonNode eventInfo = rootNode.get(eventType);
elementPath.remove(0);
if (elementPath.size() > 0) {
// Loop through elements in particular event Type and get type
for (JsonNode allOf : eventInfo) {
for (JsonNode allElements : allOf) {
Optional<String> type = recurse(allElements, new Stack<String>(), elementPath);
if (type.isPresent()) {
return type;
}
}
}
}
return Optional.empty();
}
private static Optional<String> recurse(JsonNode node, Stack<String> elements, List<String> elementPath) {
// Check if its Reference type
if (node.get("$ref") != null) {
// Check if the element has reference to other definition
final String ref = node.toString().contains("definitions") ? node.get("$ref").toString().replace("\"", "").replaceAll("#/definitions/",
"") : null;
// Check if the Reference is null
if (ref != null) {
final JsonNode refElements = _instance.outerNode.get(ref);
return recurse(refElements, elements, elementPath);
}
} else if (node.get("items") != null && elementPath.size() > 0) {
// Check if the element has Array type with items
final String ref = node.get("items").toString().contains("definitions") ? node.get("items").get("$ref").toString().replace("\"", "")
.replaceAll("#/definitions/", "") : null;
if (elementPath.size() > 0 && elementPath.get(0).equals(ref)) {
elementPath.remove(0);
final JsonNode refElements = _instance.outerNode.get(ref);
return recurse(refElements, elements, elementPath);
}
} else if (node.get("properties") != null && elementPath.size() > 0) {
// Check if the element has properties if no recurse through properties
final JsonNode prop = node.get("properties");
if (elementPath.size() > 0 && prop.get(elementPath.get(0)) != null) {
final JsonNode elementInfo = prop.path(elementPath.get(0));
elementPath.remove(0);
return recurse(elementInfo, elements, elementPath);
}
} else if (node.path("type") != null && elementPath.size() == 0) {
// If the element has type then return the final Type
return Optional.of(node.get("type").toString());
} else {
// If the element is not found then directly check
}
return Optional.empty();
}
}
推荐阅读
- javascript - 如何在具有对象的深度嵌套数组中获取最后一个孩子
- python - Kivy - 在代码中的 TabbedPanelItems 之间切换
- r - 是否可以在 igraph R 库中使用多线程?
- gcc - printf 与 utf-8 编码字符串的兼容性
- android-studio - 我想用我的 android 手机作为模拟器在 windows 7 上运行颤振。在我的调试模式下进行了调整。如何解决颤振医生命令的问题
- microsoft-graph-api - Microsoft Graph:NextLink...上一个链接呢?
- android - AppCompatDialogs 在使用 alpha 版本的各种操作系统版本上崩溃 Material Components
- python - Python 2.7 的 input() 和 raw_input() 问题。用户输入的数据未正确读取。到底是怎么回事?
- shell - sqlplus 命令在 linux 中从 crontab 执行时返回帮助页面
- r - 清理具有获得姓氏和名字的分隔空间的列,以便我可以从我的数据框中过滤它