How to compare two Json objects using C#

后端 未结 6 1818
我在风中等你
我在风中等你 2021-02-05 19:57

I have two Json objects as below need to be compared. I am using Newtonsoft libraries for Json parsing.

string InstanceExpected = jsonExpected;
string InstanceAc         


        
相关标签:
6条回答
  • 2021-02-05 20:37

    In case you are looking for a light weight library to compares two JSON objects (or practically any serializable entities) You may use the following package, it (currently) uses Newtonsoft.Json JObjects and highlights the differences based on a simple convention (* modified, - removed, + added from/ to the second operand) as shown below.

    JSON 1

    {
      "name":"John",
      "age":30,
      "cars": {
        "car1":"Ford",
        "car2":"BMW",
        "car3":"Fiat"
      }
     }
    

    JSON 2

    {
      "name":"John",
      "cars": {
        "car1":"Ford",
        "car2":"BMW",
        "car3":"Audi",
        "car4":"Jaguar"
      }
     }
    

    Usage

    
     var j1 = JToken.Parse(Read(json1));
     var j2 = JToken.Parse(Read(json2));
    
     var diff = JsonDifferentiator.Differentiate(j1,j2);
    
    

    Result

    {
      "-age": 30,
      "*cars": {
        "*car3": "Fiat",
        "+car4": "Jaguar"
      }
    }
    

    https://www.nuget.org/packages/JsonDiffer

    0 讨论(0)
  • 2021-02-05 20:38

    Made a non-recursive method which will remove twins - idea is to remove same elements from very similar JSONs, so that there will remain only different nodes in each object:

    public void RemoveTwins(ref BreadthFirst bf1, ref BreadthFirst bf2) {
        JsonNode traversal = bf1.Next();
        Boolean removed = false;
        do {
            if (!removed) {
                if (bf2.Current != null) while (bf1.Level == bf2.Level && bf2.Next() != null) ;
                if (bf2.Current != null) while (bf1.Level != bf2.Level && bf2.Next() != null) ;
                else bf2.Current = bf2.root;
            }
            else traversal = bf1.Next();
            if (bf2.Level < 0) bf2.Current = bf2.Root;
            do {
                removed = bf1.NextAs(bf1.src, bf2, bf2.src);
                if (removed && bf1.Orphan && bf2.Orphan) {
                    JsonNode same = bf1.Current.Parent;
                    traversal = bf1.RemoveCurrent();
                    same = bf2.Current.Parent;
                    bf2.RemoveCurrent();
                    bf1.UpdateLevel();
                    bf2.UpdateLevel();
                    if (traversal == null
                    || bf1.Root == null || bf2.Root == null
                    || (bf1.Level == 0 && bf1.Current.NodeBelow == null)) {
                        traversal = null;
                        break;
                    }
                } else
                if (!removed) {
                    break; 
                } else removed = false;
            } while (removed);
            if (!removed) traversal = bf1.Next();
        } while (traversal != null);
    }
    

    Complete code + parser on my GitHub (profile or below).
    Older CSV version which also sorts input mentioned in my question here How to compare big JSON's? (new one does not, so it could be very slow when one of objects has reversed order - it would be easier to sort during parsing or at least compare both neighbours of twins as first search step)

    0 讨论(0)
  • 2021-02-05 20:42

    Consider using the JToken.DeepEquals() method provided by Newtonsoft. It would look somewhat like this, regardless of which testing framework you're using:

    Console.WriteLine(JToken.DeepEquals(InstanceObjActual, InstanceObjExpected));
    // false
    
    0 讨论(0)
  • 2021-02-05 20:43

    I did a bit more digging and was able to find out why the OP's test code doesn't run as expected. I was able to fix it by installing and using the FluentAssertions.Json nuget package.

    One important thing:

    Be sure to include using FluentAssertions.Json otherwise false positives may occur.

    Test code is the following:

    using FluentAssertions;
    using FluentAssertions.Json;
    using Newtonsoft.Json;
    using Newtonsoft.Json.Linq;
    using NUnit.Framework;
    
    [TestFixture]
    public class JsonTests
    {
        [Test]
        public void JsonObject_ShouldBeEqualAsExpected()
        {
            JToken expected = JToken.Parse(@"{ ""Name"": ""20181004164456"", ""objectId"": ""4ea9b00b-d601-44af-a990-3034af18fdb1%>"" }");
            JToken actual = JToken.Parse(@"{ ""Name"": ""AAAAAAAAAAAA"", ""objectId"": ""4ea9b00b-d601-44af-a990-3034af18fdb1%>"" }");
    
            actual.Should().BeEquivalentTo(expected);
        }
    }
    

    Running the test:

    0 讨论(0)
  • 2021-02-05 20:44

    After you deserialize the json to C# object the correct way is to implement the IComparable interface in the deserialized class and compare the 2 objects.

    So:

    using System;
    using System.Collections.Generic;
    
    class MyObj : IComparable<MyObj>
    {
        public string Name { get; set; }
        public string ObjectID { get; set; }
    
        public int CompareTo(MyObj other)
        {
            if ((this.Name.CompareTo(other.Name) == 0) &&
                (this.ObjectID.CompareTo(other.ObjectID) == 0))
            {
                return 0;
            }
            return -1;
        }
    }
    
    0 讨论(0)
  • 2021-02-05 20:55

    One option is to deserialize the json strings into C# objects and compare them.

    This approach requires more work comparing to using JToken.DeepEquals (as suggested by @JessedeWit), but has the advantage of giving better error messages if your tests fail (see screenshot below).

    Your json string can be modelled into the following class:

    public class Entity
    {
        [JsonProperty("Name")]
        public string Name { get; set; }
    
        [JsonProperty("objectId")]
        public string ObjectId { get; set; }
    }
    

    In your test, deserialize the json strings into objects and compare them:

    [TestFixture]
    public class JsonTests
    {
        [Test]
        public void JsonString_ShouldBeEqualAsExpected()
        {
            string jsonExpected = @"{ ""Name"": ""20181004164456"", ""objectId"": ""4ea9b00b-d601-44af-a990-3034af18fdb1%>"" }";
            string jsonActual = @"{ ""Name"": ""AAAAAAAAAAAA"", ""objectId"": ""4ea9b00b-d601-44af-a990-3034af18fdb1%>"" }";
    
            Entity expectedObject = JsonConvert.DeserializeObject<Entity>(jsonExpected);
            Entity actualObject = JsonConvert.DeserializeObject<Entity>(jsonActual);
    
            actualObject.Should().BeEquivalentTo(expectedObject);
        }
    }
    

    PS: I used NUnit and FluentAssertions in my test method. Running the test:

    0 讨论(0)
提交回复
热议问题