Partager via


Mappage entre JSON et XML

Les lecteurs et writers produits par le JsonReaderWriterFactory fournissent une API XML sur le contenu de JavaScript Objet Notation (JSON). JSON encode des données à l'aide d'un sous-ensemble de littéraux d'objet JavaScript. Les lecteurs et writers produits par cette fabrique sont également utilisés lorsque le contenu JSON est envoyé ou reçu par les applications Windows Communication Foundation (WCF) à l’aide du WebMessageEncodingBindingElement ou du WebHttpBinding.

En cas d'initialisation avec le contenu JSON, le lecteur JSON se comporte de la même façon qu'un lecteur XML textuel sur une instance de XML. Le writer JSON, lorsqu'il reçoit une séquence d'appels qui sur un lecteur XML textuel produit une certaine instance XML, écrit un contenu JSON. Le mappage entre cette instance de XML et le contenu JSON est décrit dans cette rubrique pour une utilisation dans des scénarios avancés.

En interne, JSON est représenté comme un jeu d’informations XML lorsqu’il est traité par WCF. Il n'est pas nécessaire généralement de vous préoccuper de cette représentation interne étant donné que le mappage est uniquement un mappage logique : JSON n'est pas généralement converti physiquement en XML en mémoire ou converti vers JSON à partir du XML. Le mappage signifie que les API XML sont utilisées pour accéder au contenu JSON.

Lorsque WCF utilise JSON, le scénario habituel est que le DataContractJsonSerializer est branché automatiquement par le comportement WebScriptEnablingBehavior, ou par le comportement WebHttpBehavior le cas échéant. Le DataContractJsonSerializer comprend le mappage entre JSON et les jeux d'informations XML et fonctionne comme s'il traite directement avec JSON. (Il est possible d'utiliser le DataContractJsonSerializer avec tout lecteur ou writer XML à condition que le XML se conforme au mappage suivant).

Dans les scénarios avancés, il peut devenir nécessaire d'accéder directement au mappage suivant. Ces scénarios se produisent lorsque vous souhaitez sérialiser et désérialiser JSON de manières personnalisées, sans avoir à compter sur le DataContractJsonSerializer, ou lors d'un traitement direct avec le type Message pour les messages qui contiennent JSON. Le mappage JSON-XML est également utilisé pour l'enregistrement des messages. Lors de l’utilisation de la fonctionnalité d’enregistrement des messages dans WCF, les messages JSON sont enregistrés au format XML selon le mappage décrit dans la section suivante.

Pour clarifier le concept d'un mappage, l'exemple suivant est celui d'un document JSON.

{"product":"pencil","price":12}

Pour lire ce document JSON à l'aide de l'un des lecteurs évoqués précédemment, utilisez la même séquence d'appels XmlDictionaryReader que pour lire le document XML suivant.

<root type="object">
    <product type="string">pencil</product>
    <price type="number">12</price>
</root>

En outre, si le message JSON dans l’exemple est reçu par WCF et enregistré, vous pouvez consulter le fragment XML dans le journal précédent.

Mappage entre JSON et le jeu d'informations XML

De manière formelle, le mappage se fait entre JSON, comme décrit dans la RFC 4627 (sauf avec certaines restrictions assouplies et d’autres restrictions ajoutées) et le jeu d’informations XML (et pas XML textuel), comme décrit dans le jeu d’informations XML (page pouvant être en anglais). Consultez cette rubrique pour les définitions des éléments d’informations et des champs entre [crochets].

Un document JSON vide est mappé aux documents XML vierges, et un document XML vierge est mappé à un document JSON vierge. Sur le mappage XML à JSON, des espaces blanc précédents et des caractères d’espace blanc de fin après le document ne sont pas autorisés.

Le mappage est défini entre un élément DII (Document Information Item) ou un élément EII (Element Information Item) et JSON. L'élément EII, ou la propriété de l'élément DII [élément de document] est appelé l'élément JSON racine. Notez que les fragments de document (XML avec plusieurs éléments racine) ne sont pas pris en charge dans ce mappage.

Exemple : Document suivant :

<?xml version="1.0"?>
<root type="number">42</root>

Et les éléments suivants :

<root type="number">42</root>

Ces deux éléments ont un mappage à JSON. L'élément <root> est l’élément racine JSON dans les deux cas.

De plus, les éléments suivants doivent être pris en compte dans le cas d'un élément DII :

  • Certains éléments dans la liste [enfants] ne doivent pas être présents. Ne comptez pas sur ce fait pour la lecture du XML mappé à partir de JSON.

  • La liste [enfants] ne contient aucun élément d'informations de commentaire.

  • La liste [enfants] ne contient aucun élément d'informations DTD.

  • La liste [enfants] ne contient aucun élément d’informations PI (Personal Informations) (la déclaration <?xml…> n’est pas considérée comme un élément d’informations PI)

  • Le jeu [notations] est vide.

  • Le jeu [entités non analysées] est vide.

Exemple : le document suivant n'a aucun mappage à JSON parce que [enfants] contient un PI et un commentaire.

<?xml version="1.0"?>
<!--comment--><?pi?>
<root type="number">42</root>

L'EII pour l'élément JSON racine a les caractéristiques suivantes :

  • [nom local] a la valeur "racine".

  • [nom d'espace de noms] n'a aucune valeur.

  • [préfixe] n'a aucune valeur.

  • [enfants] peut contenir des éléments EII (qui représentent des éléments internes décrits plus loin) ou des éléments CII (éléments d'informations de caractère décrits plus loin) ou aucun d'entre eux, mais pas les deux.

  • [attributs] peut contenir les éléments AII (Attribute Information Items) facultatifs suivants

  • L'attribut de type JSON ("type") décrit plus loin. Cet attribut est utilisé pour conserver le type JSON (chaîne, nombre, booléen, objet, tableau ou null) dans le XML mappé.

  • L'attribut de nom de contrat de données ("__type") décrit plus loin. Cet attribut peut être présent uniquement si l'attribut de type JSON est aussi présent et sa [valeur normalisée] est "objet." Cet attribut est utilisé par le DataContractJsonSerializer pour conserver des informations de type de contrat de données - par exemple, dans les cas polymorphes où un type dérivé est sérialisé et où un type de base est attendu. Si vous n'utilisez le DataContractJsonSerializer, dans la plupart des cas, cet attribut est ignoré.

  • [espaces de noms dans l’étendue] contient la liaison de « xml » vers http://www.w3.org/XML/1998/namespace comme exigé par la spécification infoset.

  • [enfants], [attributs] et [espaces de noms dans l'étendue] ne doivent pas avoir d'éléments autres que ceux spécifiés précédemment et [attributs d'espace de noms] ne doit pas avoir de membres, mais ne compte pas sur ces faits pour la lecture du XML mappé à partir de JSON.

Exemple : le document suivant n'a aucun mappage à JSON parce que [attributs d'espace de noms] n'est pas vide.

<?xml version="1.0"?>
<root xmlns:a="myattributevalue">42</root>

L'élément AII pour l'attribut de type JSON a les caractéristiques suivantes :

  • [nom d'espace de noms] n'a aucune valeur.
  • [préfixe] n'a aucune valeur.
  • [nom local] est "type".
  • [valeur normalisée] est une des valeurs de type possibles décrite dans la section suivante.
  • [spécifié] a la valeur true.
  • [type d'attribut] n'a aucune valeur.
  • [références] n'a aucune valeur.

L'élément AII pour l'attribut de nom de contrat de données a les caractéristiques suivantes :

  • [nom d'espace de noms] n'a aucune valeur.
  • [préfixe] n'a aucune valeur.
  • [nom local] est "__type (deux traits de soulignement puis "type").
  • [valeur normalisée] est toute chaîne Unicode valide. Le mappage de cette chaîne à JSON est décrit dans la section suivante.
  • [spécifié] a la valeur true.
  • [type d'attribut] n'a aucune valeur.
  • [références] n'a aucune valeur.

Les éléments internes contenus dans l'élément JSON racine ou d'autres éléments internes ont les caractéristiques suivantes :

  • [nom local] peut avoir toute valeur décrite plus loin.
  • [nom d'espace de noms], [préfixe], [enfants], [attributs], [attributs d'espace de noms], et [espaces de noms dans l'étendue] sont soumis aux mêmes règles que l'élément JSON racine.

Dans l'élément JSON racine et les éléments internes, l'attribut de type JSON définit le mappage à JSON et les [enfants] possibles et leur interprétation. La [valeur normalisée] de l’attribut respecte la casse et doit être en minuscule et ne peut pas contenir d’espace blanc.

[valeur normalisée] de l’attribut AII de type JSON [enfants] autorisés de l'élément EII correspondant Mappage à JSON
string (ou absence de l'élément AII de type JSON)

Un string et l'absence de l'élément AII de type JSON sont les mêmes que la valeur par défaut string.

Ainsi, <root> string1</root> mappe à "string1" string JSON.
0 ou plusieurs éléments CII JSON string (JSON RFC, section 2.5). Chaque char est un caractère qui correspond au [code de caractère] du CII. En l'absence d'élément CII, il mappe à un string JSON vide.

Exemple : l'élément suivant mappe à un fragment JSON :

<root type="string">42</root>

Le fragment JSON est "42".

Sur le mappage XML à JSON, les caractères qui doivent faire l'objet d'une séquence d'échappement sont mappés aux caractères d'échappement, tous les autres sont mappés aux caractères de non échappement. Le caractère "/" est spécial. Il fait l'objet d'un échappement même si cela n'est pas obligatoire (écrit comme "\/").

Exemple : l'élément suivant mappe à un fragment JSON.

<root type="string">the "da/ta"</root>

Le fragment JSON est "the \"da\/ta \".

Sur le mappage JSON à XML, les caractères échappés et les caractères non échappés ne sont pas correctement mappés au [code de caractère] correspondant.

Exemple : le fragment JSON "\u0041BC" mappe à l'élément XML suivant.

<root type="string">ABC</root>

La chaîne peut être entourée par des espace blanc ('ws' dans la section 2 de la RFC JSON) qui n’est pas mappée au code XML.

Exemple : Le JSON fragmente "ABC", (il y a des espaces avant le premier guillemet double) mappe à l’élément XML suivant.

<root type="string">ABC</root>

Tout espace blanc dans XML est mappé à l’espace blanc dans JSON.

Exemple : l'élément XML suivant mappe à un fragment JSON.

<root type="string"> A BC </root>

Le fragment JSON est " A BC ".
number 1 ou plusieurs éléments CII Un JSON number (JSON RFC, section 2.4), peut-être entouré par un espace blanc. Chaque caractère dans la combinaison nombre/espace blanc est un caractère qui correspond au [code de caractère] à partir de l’élément CII.

Exemple : l'élément suivant mappe à un fragment JSON.

<root type="number"> 42</root>

Le fragment JSON est 42.

(Les espaces blancs sont conservés.)
boolean 4 ou 5 éléments CII (qui correspondent à la valeur true ou false), peuvent être entourés par un espace blanc CII supplémentaire. Une séquence CII qui correspond à la chaîne "true" est mappée au littéral true, et une séquence CII qui correspond à la chaîne "false" est mappée au littéral false. L’espace blanc encadrant est conservé.

Exemple : l'élément suivant mappe à un fragment JSON.

<root type="boolean"> false</root>

Le fragment JSON is false.
null Aucun autorisé. Littéral null. Sur le mappage JSON à XML, null peut être entouré par un espace blanc ('ws' dans section 2) qui n’est pas mappé au XML.

Exemple : l'élément suivant mappe à un fragment JSON.

<root type="null"/>

ou

<root type="null"></root>

:

Le fragment JSON dans les deux cas est Null.
object 0 ou plusieurs éléments EII. Un begin-object (accolade ouvrante) comme dans la section 2.2 de la FRC JSON, suivi d'un enregistrement membre pour chaque élément EII décrit plus loin. En présence de plusieurs éléments EII, il y a des séparateurs de valeur (virgules) entre les enregistrements membre. Cet ensemble est suivi par un objet de fin (accolade fermante).

Exemple : l'élément suivant mappe au fragment JSON.

<root type="object">

<type1 type="string">aaa\</type1>

<type2 type="string">bbb\</type2>

</root >

Le fragment JSON is {"type1":"aaa","type2":"bbb"}.

Si l'attribut de type de contrat de données est présent sur le mappage XML à JSON, tout membre supplémentaire est inséré au début. Son nom est le [nom local] de l'attribut de type de contrat de données ("__type"), et sa valeur est la [valeur normalisée] de l'attribut. Inversement, sur le mappage JSON à XML, si le premier nom de l'enregistrement membre est le [nom local] de l'attribut de type de contrat de données (autrement dit, "__type"), un attribut de type de contrat de données correspondant est présent dans le XML mappé, mais un élément EII correspondant n'est pas présent. Notez que cet enregistrement membre doit se produire en premier dans l'objet JSON pour que ce mappage spécial s'applique. Il s'agit d'une nouveauté par rapport au traitement JSON habituel dans lequel l'ordre des enregistrements membres n'est pas significatif.

Exemple :

Le fragment JSON suivant mappe au XML.

{"__type":"Person","name":"John"}

Le XML est le code suivant.

<root type="object" __type="Person"> <name type="string">John</name> </root>

Remarquez que l'élément AII __type est présent, mais il n'y a aucun élément EII __type.

Toutefois, si l'ordre dans le JSON est inversé comme indiqué dans l'exemple suivant.

{"name":"John","\_\_type":"Person"}

Le XML correspondant est indiqué.

<root type="object"> <name type="string">John</name> <__type type="string">Person</__type> </root>

Autrement dit, __type cesse d'avoir une signification spéciale et mappe à un élément EII, pas un élément AII.

Les règles d'échappement ou de non échappement pour l'élément AII [valeur normalisée] en cas de mappage à une valeur JSON sont les mêmes que pour les chaînes JSON spécifiées dans la ligne "chaîne" de ce tableau.

Exemple :

<root type="object" __type="\abc" />

à l'exemple précédent peut être mappé au JSON suivant.

{"__type":"\\abc"}

Sur un mappage XML au JSON, le premier élément EII [nom local] ne doit pas être "__type".

L’espace blanc (ws) n’est jamais généré sur le mappage XML à JSON pour les objets et est ignoré sur le mappage JSON à XML.

Exemple : le fragment JSON suivant mappe à un élément XML.

{ "ccc" : "aaa", "ddd" :"bbb"}

L'élément XML est indiqué dans le code suivant.

<root type="object"> <ccc type="string">aaa</ccc> <ddd type="string">bbb</bar> </root >
tableau 0 ou plusieurs éléments EII Un tableau ouvrant (crochet ouvrant) comme dans la section 2.3 de la RFC JSON suivi d'un enregistrement de tableau pour chaque élément EII comme cela est décrit plus loin. En présence de plusieurs éléments EII, il y a des séparateurs de valeur (virgules) entre les enregistrements de tableau. L'ensemble est suivi d'un tableau fermant.

Exemple : l'élément XML suivant mappe à un fragment JSON.

<root type="array"/> <item type="string">aaa</item> <item type="string">bbb</item> </root >

Le fragment JSON est ["aaa","bbb"].

L’espace blanc (ws) n’est jamais généré sur le mappage XML à JSON pour les tableaux et est ignoré sur le mappage JSON à XML.

Exemple : fragment JSON.

["aaa", "bbb"]

L'élément XML auquel il mappe.

<root type="array"/> <item type="string">aaa</item> <item type="string">bbb</item> </root >

Les enregistrements membres fonctionnent comme suit :

  • Le [nom local] de l'élément interne mappe à la partie string du member défini dans la section 2.2 de la RFC JSON.

Exemple : l'élément suivant mappe à un fragment JSON.

<root type="object">
    <myLocalName type="string">aaa</myLocalName>
</root>

Le fragment JSON suivant est affiché.

{"myLocalName":"aaa"}
  • Sur le mappage XML à JSON, les caractères qui doivent faire l'objet d'une séquence d'échappement dans JSON le sont, et tous les autres ne le sont pas. Le caractère "/", même s'il ne s'agit pas d'un caractère qui doit faire l'objet d'une séquence d'échappement, est néanmoins échappé (ce qui n'est pas nécessaire sur le mappage JSON à XML). Cela est requis pour prendre en charge le format ASP.NET AJAX pour les données DateTime dans JSON.

  • Sur le mappage JSON au XML, tous les caractères (y compris les caractères non échappés, si nécessaire) sont acceptés pour former un string qui produit un [nom local].

  • Les éléments internes [enfants] mappent à la valeur dans section 2.2, selon JSON Type Attribute comme pour Root JSON Element. Plusieurs niveaux d'imbrication des éléments EII (y compris l'imbrication dans des tableaux) sont autorisés.

Exemple : l'élément suivant mappe à un fragment JSON.

<root type="object">
    <myLocalName1 type="string">myValue1</myLocalName1>
    <myLocalName2 type="number">2</myLocalName2>
    <myLocalName3 type="object">
        <myNestedName1 type="boolean">true</myNestedName1>
        <myNestedName2 type="null"/>
    </myLocalName3>
</root >

Le fragment JSON suivant est celui auquel il est mappé.

{"myLocalName1":"myValue1","myLocalName2":2,"myLocalName3":{"myNestedName1":true,"myNestedName2":null}}

Notes

Il n'y a aucune étape d'encodage XML dans le mappage précédent. Par conséquent, WCF prend en charge seulement des documents JSON où tous les caractères dans les noms clé sont des caractères valides dans les noms d’élément XML. Par exemple, le document JSON {"<":"a"} n’est pas pris en charge parce que < n’est pas un nom valide pour un élément XML.

La situation inverse (caractères valide dans XML mais pas dans JSON) ne provoque pas de problèmes étant donné que le mappage précédent inclut des étapes d'échappement et de non échappement JSON.

Les enregistrements du tableau fonctionnent comme suit :

  • Le [nom local] de l'élément interne est "élément."

  • [enfants] de l'élément interne mappe à la valeur dans la section 2.3, selon l'attribut de type JSON comme pour l'élément JSON racine. Plusieurs niveaux d'imbrication des éléments EII (y compris l'imbrication dans des objets) sont autorisés.

Exemple : l'élément suivant mappe à un fragment JSON.

<root type="array">
    <item type="string">myValue1</item>
    <item type="number">2</item>
    <item type="array">
    <item type="boolean">true</item>
    <item type="null"/></item>
</root>

Les éléments suivants sont le fragment JSON.

["myValue1",2,[true,null]]

Voir aussi