有没有办法克隆一个实例org.json.JSONObject
而不对其进行字符串化并重新分析结果?
浅拷贝是可以接受的。
最简单(但速度极慢且效率低下)的方法
JSONObject clone = new JSONObject(original.toString());
使用public JSONObject(JSONObject jo, java.lang.String[] names)
构造函数和public static java.lang.String[] getNames(JSONObject jo)
方法。
JSONObject copy = new JSONObject(original, JSONObject.getNames(original));
由于无法在android$JSONObject.getNames(original)
中访问,您可以使用以下方法:
public JSONObject shallowCopy(JSONObject original) {
JSONObject copy = new JSONObject();
for ( Iterator<String> iterator = original.keys(); iterator.hasNext(); ) {
String key = iterator.next();
JSONObject value = original.optJSONObject(key);
try {
copy.put(key, value);
} catch ( JSONException e ) {
//TODO process exception
}
}
return copy;
}
但请记住,它不是深拷贝。
对于Android开发者来说,不使用的最简单的解决方案.getNames
是:
JSONObject copy = new JSONObject();
for (Object key : original.keySet()) {
Object value = original.get(key);
copy.put(key, value);
}
注意:这只是一个浅拷贝
找不到 com.google.gwt.json.client.JSONObject 的现有深度克隆方法,但实现应该是几行代码,例如:
public static JSONValue deepClone(JSONValue jsonValue){
JSONString string = jsonValue.isString();
if (string != null){return new JSONString(string.stringValue());}
JSONBoolean aBoolean = jsonValue.isBoolean();
if (aBoolean != null){return JSONBoolean.getInstance(aBoolean.booleanValue());}
JSONNull aNull = jsonValue.isNull();
if (aNull != null){return JSONNull.getInstance();}
JSONNumber number = jsonValue.isNumber();
if (number!=null){return new JSONNumber(number.doubleValue());}
JSONObject jsonObject = jsonValue.isObject();
if (jsonObject!=null){
JSONObject clonedObject = new JSONObject();
for (String key : jsonObject.keySet()){
clonedObject.put(key, deepClone(jsonObject.get(key)));
}
return clonedObject;
}
JSONArray array = jsonValue.isArray();
if (array != null){
JSONArray clonedArray = new JSONArray();
for (int i=0 ; i < array.size() ; ++i){
clonedArray.set(i, deepClone(array.get(i)));
}
return clonedArray;
}
throw new IllegalStateException();
}
*注意: *我还没有测试过!
我发现的最快+最小的方法就是这个。它做深拷贝。
JSONObject 克隆=新 JSONObject(original.toMap());
我知道提问者说
浅拷贝是可以接受的。但我认为这不排除解决方案是否会进行深度复制。
更新:toMap() 函数在 Android 中不可用。但是在 maven 上 groupId org.json 下可用的 org.json 库有它:https ://search.maven.org/artifact/org.json/json/20210307/bundle
如果有人来这里寻找 org.google.gson 的深层克隆,因为他们没有公开他们的 deepClone() 方法,这就是我想出的......
public static JsonElement deepClone(JsonElement el){
if (el.isJsonPrimitive() || el.isJsonNull())
return el;
if (el.isJsonArray()) {
JsonArray array = new JsonArray();
for(JsonElement arrayEl: el.getAsJsonArray())
array.add(deepClone(arrayEl));
return array;
}
if(el.isJsonObject()) {
JsonObject obj = new JsonObject();
for (Map.Entry<String, JsonElement> entry : el.getAsJsonObject().entrySet()) {
obj.add(entry.getKey(), deepClone(entry.getValue()));
}
return obj;
}
throw new IllegalArgumentException("JsonElement type " + el.getClass().getName());
}
这里有一些方法可以合并两个 JsonObject
public static JsonObject merge(String overrideJson, JsonObject defaultObj) {
return mergeInto((JsonObject)new JsonParser().parse(overrideJson), defaultObj);
}
public static JsonObject merge(JsonObject overrideObj, JsonObject defaultObj) {
return mergeOverride((JsonObject)deepClone(defaultObj), overrideObj);
}
public static JsonObject mergeOverride(JsonObject targetObj, JsonObject overrideObj) {
for (Map.Entry<String, JsonElement> entry : overrideObj.entrySet())
targetObj.add(entry.getKey(), deepClone(entry.getValue()));
return targetObj;
}
public static JsonObject mergeInto(JsonObject targetObj, JsonObject defaultObj) {
for (Map.Entry<String, JsonElement> entry : defaultObj.entrySet()) {
if (targetObj.has(entry.getKey()) == false)
targetObj.add(entry.getKey(), deepClone(entry.getValue()));
}
return targetObj;
}