11 votes

Le C# dispose-t-il d'une bibliothèque pour analyser le JSON en cascade à plusieurs niveaux ?

Existe-t-il une bibliothèque (C# de préférence) pour résoudre ce que j'appellerais le JSON en cascade à plusieurs niveaux ?

Voici un exemple de ce que je veux dire : (Pseudocode/C#)

var json1 = @"{
    ""firstName"": ""John"",
    ""lastName"": ""Smith""
    }";

var json2 = @"{
    ""firstName"": ""Albert""
    }";

var json3 = @"{
    ""phone"": ""12345""
    }";

var cascadingJSON = JSON.Cascade(json1, json2, json3);

Résultat (Même comportement que le CSS)

{
    "firstName"": "Albert", /*Overridden*/
    "lastName"": "Smith", /*Inherited*/
    "phone"": "12345"   }"; /*Added*/
}

Edit 1 - Exemple plus complexe

const string json1 =
                @"{
                     ""firstName"": ""John"",
                     ""lastName"": ""Smith"",
                     ""age"": 25,
                     ""address"": 
                     {
                         ""streetAddress"": ""21 2nd Street"",
                         ""city"": ""New York"",
                         ""state"": ""NY"",
                         ""postalCode"": ""10021""
                     },
                     ""phoneNumber"": 
                     [
                         {
                           ""type"": ""home"",
                           ""number"": ""212 555-1234""
                         },
                         {
                           ""type"": ""fax"",
                           ""number"": ""646 555-4567""
                         }
                     ]
                 }";

            const string json2 =
                @"{
                     ""firstName"": ""John2"",
                     ""lastName"": ""robert"",
                     ""age"": 25,
                     ""address"": 
                     {
                         ""state"": ""FL"",
                     },
                     ""phoneNumber"": 
                     [
                         {
                           ""type"": ""fax"",
                           ""number"": ""222 222-2222""
                         },
                         {
                           ""type"": ""iphone"",
                           ""number"": ""111 111-1111""
                         }
                     ]
                 }";

            const string json3 =
                @"{
                     ""firstName"": ""John3"",
                     ""father"": ""guy""
                 }";

            const string expectedResult =
                @"{
                     ""firstName"": ""John3"",
                     ""lastName"": ""robert"",
                     ""age"": 25,
                     ""father"": ""guy"",
                     ""address"": 
                     {
                         ""streetAddress"": ""21 2nd Street"",
                         ""city"": ""New York"",
                         ""state"": ""FL"",
                         ""postalCode"": ""10021""
                     },
                     ""phoneNumber"": 
                     [
                         {
                           ""type"": ""home"",
                           ""number"": ""212 555-1234""
                         },
                         {
                           ""type"": ""fax"",
                           ""number"": ""222 222-2222""
                         },
                         {
                           ""type"": ""iphone"",
                           ""number"": ""111 111-1111""
                         }
                     ]
                 }";

Edit 2

Après avoir réfléchi un peu plus aux exigences, je vois que l'exemple le plus complexe ne pourrait jamais fonctionner tel quel. La fonction Cascading ne serait pas en mesure de savoir si, par exemple, un certain numéro de téléphone a été modifié ou s'il s'agit d'un nouveau numéro. Pour que cela fonctionne, chaque sous-entité doit avoir un identifiant unique.

18voto

Greg Points 11248

C'est super facile en utilisant l'excellent Bibliothèque JSON.NET . Cette méthode combine des objets dont les propriétés sont des chaînes de caractères, des nombres ou des objets.

public static string Cascade(params string[] jsonArray)
{
    JObject result = new JObject();
    foreach (string json in jsonArray)
    {
        JObject parsed = JObject.Parse(json);
        foreach (var property in parsed)
            result[property.Key] = property.Value;
    }
    return result.ToString();
}

Résultat, compte tenu de votre exemple :

{
  "firstName": "Albert",
  "lastName": "Smith",
  "phone": "12345"
}

Modification en réponse à votre question mise à jour :

En adaptant cette solution pour qu'elle fonctionne de manière récursive, vous pouvez fusionner les objets enfants. L'exemple suivant correspondra aux résultats attendus (sauf pour le tableau). Vous pourrez facilement étendre cette solution pour fusionner des tableaux ( JArray ) d'une manière similaire à la façon dont il fusionne les objets ( JObject ).

public static string Cascade(params string[] jsonArray)
{
    JObject result = new JObject();
    foreach (string json in jsonArray)
    {
        JObject parsed = JObject.Parse(json);
        Merge(result, parsed);
    }
    return result.ToString();
}

private static void Merge(JObject receiver, JObject donor)
{
    foreach (var property in donor)
    {
        JObject receiverValue = receiver[property.Key] as JObject;
        JObject donorValue = property.Value as JObject;
        if (receiverValue != null && donorValue != null)
            Merge(receiverValue, donorValue);
        else
            receiver[property.Key] = property.Value;
    }
}

2voto

Matthew Flaschen Points 131723

Pas que je sache. Pour le cas simple, vous pouvez utiliser n'importe quelle bibliothèque JSON, puis fusionner les dictionnaires avec une solution telle que celui-ci . Par exemple, en utilisant Newtonsoft/Json.NET :

Dictionary<String, String> dict1, dict2, dict3, merged;
dict1 = JsonConvert.DeserializeObject<Dictionary<string,string>>(json1);
dict2 = JsonConvert.DeserializeObject<Dictionary<string,string>>(json2);
dict3 = JsonConvert.DeserializeObject<Dictionary<string,string>>(json3);
merged = Merge(new[]{dict1, dict2, dict3});

Évidemment, dans un code de production, il faut éliminer les lignes redondantes.

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X