首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >深入比较两种json并展示其差异

深入比较两种json并展示其差异
EN

Stack Overflow用户
提问于 2016-12-26 19:40:30
回答 1查看 16.8K关注 0票数 2

我正在尝试比较两个动态json数据,如果它们不相等,则打印差异。为此,我使用了

代码语言:javascript
复制
Type mapType = new TypeToken<Map<String, Object>>(){}.getType();
Map<String,Object> firstMap = g.fromJson(jsonElement1, mapType);
Map<String, Object> secondMap = g.fromJson(jsonElement2, mapType);
System.out.println(Maps.difference(firstMap, secondMap));

我需要在控制台中显示差异。这是我试图在其中显示json的差异的java代码。

代码语言:javascript
复制
      public class Tester {
public static ArrayList<Object> ls1 = new ArrayList<Object>();
 public static              ArrayList<Object> ls2 = new ArrayList<Object>();
            public static void main(String[] args) throws Exception {
                JsonParser parser = new JsonParser();

                try{
                    Gson g = new Gson();
                    JsonElement jsonElement1 = parser
                            .parse(new FileReader("D:\\file1.json"));
                    JsonElement jsonElement2 = parser
                            .parse(new FileReader("D:\\file2.json"));
                    System.out.println("Is the two JSON File Same: "+compareJson(jsonElement1, jsonElement2));
                    if(!compareJson(jsonElement1, jsonElement2)){
                        Type mapType = new TypeToken<Map<String, Object>>(){}.getType();
                        Map<String,Object> firstMap = g.fromJson(jsonElement1, mapType);
                        Map<String, Object> secondMap = g.fromJson(jsonElement2, mapType);
                        System.out.println(Maps.difference(firstMap, secondMap));
                    }
                    else{
                        System.out.println("The Two JSON Are SAME!!!!!!!!!!!!!!!");
                    }

                }catch(Exception e1){
                    e1.printStackTrace();
                }

            }

            public static boolean compareJson(JsonElement json1, JsonElement json2) {
                boolean isEqual = true;

                // Check whether both jsonElement are not null
                if (json1 != null && json2 != null) {

                    // Check whether both jsonElement are objects
                    if (json1.isJsonObject() && json2.isJsonObject()) {
                        Set<Entry<String, JsonElement>> ens1 = ((JsonObject) json1).entrySet();
                        Set<Entry<String, JsonElement>> ens2 = ((JsonObject) json2).entrySet();
                        JsonObject json2obj = (JsonObject) json2;
                        if (ens1 != null && ens2 != null && (ens2.size() == ens1.size())) {
                            // Iterate JSON Elements with Key values
                            for (Entry<String, JsonElement> en : ens1) {
                                isEqual = isEqual && compareJson(en.getValue(),json2obj.get(en.getKey()));
                            }
                        } else {
                            return false;
                        }
                    }
                    // Check whether both jsonElement are arrays
                    else if (json1.isJsonArray() && json2.isJsonArray()) {
                        JsonArray jarr1 = json1.getAsJsonArray();
                        JsonArray jarr2 = json2.getAsJsonArray();
                        if (jarr1.size() != jarr2.size()) {
                            return false;
                        } else {
                            int i = 0;
                            // Iterate JSON Array to JSON Elements
                            for (JsonElement je : jarr1) {
                                isEqual = isEqual && compareJson(je, jarr2.get(i));
                                i++;
                            }
                            if (isEqual) {
                                Object[] o1 = ls1.toArray();
                                Object[] o2 = ls2.toArray();
                                isEqual = Arrays.deepEquals(o1, o2);
                            }
                        }

                    }

                    // Check whether both jsonElement are null
                    else if (json1.isJsonNull() && json2.isJsonNull()) {
                        return true;
                    }

                    // Check whether both jsonElement are primitives
                    else if (json1.isJsonPrimitive() && json2.isJsonPrimitive()) {
                        ls1.add(json1);
                        ls2.add(json2);         
                    }               
                } else if (json1 == null && json2 == null) {
                    return true;
                } else {
                    return false;
                }
                return isEqual;
            }
        }

对象数组

代码语言:javascript
复制
if (isEqual) {
                            Object[] o1 = ls1.toArray();
                            Object[] o2 = ls2.toArray();
                            isEqual = Arrays.deepEquals(o1, o2);
                        }

对象数组o1和o2为null不知道为什么它为null。这是我用来测试公司的两个示例json在这两个json中,json1的值是不同的,所以代码必须返回false,但代码返回true

EN

回答 1

Stack Overflow用户

发布于 2016-12-28 16:56:42

评论部分太长了,所以这里有个答案:

首先,如果您的Json对象具有不同的结构,您仍然会遇到一个问题。例如:

代码语言:javascript
复制
{"Company List":["Compnay: eBay","Compnay: Google","Compnay: Paypal"]} 

{"Company List":"ImAString"}

两者都不是null,它是一个原语和一个数组。您不会捕捉到这种情况,因此您的isEqual将保持为真。

无论如何,回到手头的问题。

在你的例子中,数组的排序不应该是相同的。

代码语言:javascript
复制
if (isEqual) {
    Object[] o1 = ls1.toArray();
    Object[] o2 = ls2.toArray();
    isEqual = Arrays.deepEquals(o1, o2);
}

通过将其更改为以下代码,您应该能够轻松地解决此问题:

代码语言:javascript
复制
if (isEqual) {
    isEqual = ls1.containsAll(ls2);
}

尽管我应该提一下,我发现你的ls1和ls2都是静态的,并且一直在获取条目,这真的很奇怪。在你的示例中,这并不重要,因为它们太小了,但是在大文件中,你会得到巨大的ArrayLists,其中有很多值多次出现。

每次使用JsonArray时,将它们都进行比较似乎是一种巨大的时间浪费。

由于这仍然存在无法比较排序不同的JSON数组与非原语的JSON元素的问题,因此这里是比较两个数组的强力方法。请记住,这远不是一种最优的方法,但它相对容易实现:

代码语言:javascript
复制
public static boolean equality(ArrayList<Object> list1, ArrayList<Object> list2) {  
    if (list1.length != list2.length) return false;

    for (Object entry1 : list1) {
        Object equalEntry = null;

        for(Object entry2 : list2) { 
            if (entry1.equals(entry2)) {
                equalEntry = entry2;
                break;
            }
        } 

        if (equalEntry == null) return false;       
        list2.remove(equalEntry);
    } 

    return true;
}

我还应该提一下,它显然不像if (entry1.equals(entry2))那么简单。此时,您可能必须执行递归并调用compare方法。此外,我很确定您的列表包含JSON元素,而不是对象;)

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/41330715

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档