我正在使用 JSON 并面临一些问题。
我想在 JSON 对象中插入/更新路径。如果路径不存在,它将被创建,然后我插入一个新值。如果它退出,它将被更新为新值
例如,我想添加这样的新路径:
val doc = JsonPath.parse(jsonString)
doc.add("$.user.name", "John")
但我总是收到此错误,因为路径不存在:
class com.jayway.jsonpath.PathNotFoundException:路径 $['user'] 中缺少属性
因此,如果路径不存在,我想创建一个新路径。
这是我的代码,但
jsonString
没有改变:
var jsonString = "{}" val conf = Configuration.defaultConfiguration().addOptions(Option.DEFAULT_PATH_LEAF_TO_NULL).addOptions(Option.SUPPRESS_EXCEPTIONS)
JsonPath.using(conf).parse(jsonString).set(JsonPath.compile("$.user.name"), "John")
Log.d("TAG", "new json = $jsonString")
请给我你的建议。非常感谢!!
我尝试了三个支持 JsonPath/JsonPointer 的不同 JSON 库(Jackson、JsonPath 和 JSON-P),但在缺少父节点的情况下,它们都无法重建 JSON 对象层次结构。因此,我想出了自己的解决方案,使用 Jackson/JsonPointer 向 JSON 对象添加新值,因为它允许浏览 JsonPointer 部分。
private static final ObjectMapper mapper = new ObjectMapper();
public void setJsonPointerValue(ObjectNode node, JsonPointer pointer, JsonNode value) {
JsonPointer parentPointer = pointer.head();
JsonNode parentNode = node.at(parentPointer);
String fieldName = pointer.last().toString().substring(1);
if (parentNode.isMissingNode() || parentNode.isNull()) {
parentNode = StringUtils.isNumeric(fieldName) ? mapper.createArrayNode() : mapper.createObjectNode();
setJsonPointerValue(parentPointer, parentNode); // recursively reconstruct hierarchy
}
if (parentNode.isArray()) {
ArrayNode arrayNode = (ArrayNode) parentNode;
int index = Integer.valueOf(fieldName);
// expand array in case index is greater than array size (like JavaScript does)
for (int i = arrayNode.size(); i <= index; i++) {
arrayNode.addNull();
}
arrayNode.set(index, value);
} else if (parentNode.isObject()) {
((ObjectNode) parentNode).set(fieldName, value);
} else {
throw new IllegalArgumentException("`" + fieldName + "` can't be set for parent node `"
+ parentPointer + "` because parent is not a container but " + parentNode.getNodeType().name());
}
}
用途:
ObjectNode rootNode = mapper.createObjectNode();
setJsonPointerValue(rootNode, JsonPointer.compile("/root/array/0/name"), new TextNode("John"));
setJsonPointerValue(rootNode, JsonPointer.compile("/root/array/0/age"), new IntNode(17));
setJsonPointerValue(rootNode, JsonPointer.compile("/root/array/4"), new IntNode(12));
setJsonPointerValue(rootNode, JsonPointer.compile("/root/object/num"), new IntNode(81));
setJsonPointerValue(rootNode, JsonPointer.compile("/root/object/str"), new TextNode("text"));
setJsonPointerValue(rootNode, JsonPointer.compile("/descr"), new TextNode("description"));
System.out.println(mapper.writerWithDefaultPrettyPrinter().writeValueAsString(rootNode));
这会生成并打印以下 JSON 对象:
{
"root" : {
"array" : [ {
"name" : "John",
"age" : 17
}, null, null, null, 12 ],
"object" : {
"num" : 81,
"str" : "text"
}
},
"descr" : "description"
}
当然,这并不能涵盖所有极端情况,但适用于大多数情况。希望这对其他人有帮助。
要创建新节点,请尝试在由 JsonPath.parse(jsonString) 结果实现的 WriteContext 接口上 put(path, key, object)。
您可以按照以下方式进行:
JsonPath.parse(jsonString).set(JsonPath.compile("$.user.name"), "John");
Put 将帮助您添加或更新密钥:
JsonPath.parse(jsonString).put("user", "name", "John");