50

有没有办法克隆一个实例org.json.JSONObject而不对其进行字符串化并重新分析结果?

浅拷贝是可以接受的。

4

7 回答 7

86

最简单(但速度极慢且效率低下)的方法

JSONObject clone = new JSONObject(original.toString());
于 2013-08-24T11:32:15.147 回答
60

使用public JSONObject(JSONObject jo, java.lang.String[] names)构造函数和public static java.lang.String[] getNames(JSONObject jo)方法。

JSONObject copy = new JSONObject(original, JSONObject.getNames(original));
于 2012-10-09T23:30:21.517 回答
3

由于无法在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;
}

但请记住,它不是深拷贝。

于 2017-04-04T11:35:48.670 回答
3

对于Android开发者来说,不使用的最简单的解决方案.getNames是:

JSONObject copy = new JSONObject();
for (Object key : original.keySet()) {
  Object value = original.get(key);
  copy.put(key, value);
}

注意:这只是一个拷贝

于 2019-08-13T01:18:01.063 回答
2

找不到 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();
}

*注意: *我还没有测试过!

于 2013-06-04T10:02:26.747 回答
0

我发现的最快+最小的方法就是这个。它做深拷贝。

JSONObject 克隆=新 JSONObject(original.toMap());

我知道提问者说

浅拷贝是可以接受的。但我认为这不排除解决方案是否会进行深度复制。

更新:toMap() 函数在 Android 中不可用。但是在 maven 上 groupId org.json 下可用的 org.json 库有它:https ://search.maven.org/artifact/org.json/json/20210307/bundle

于 2021-08-13T15:05:12.603 回答
-2

如果有人来这里寻找 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;
}
于 2016-10-05T02:54:35.620 回答