AspNetCore.Docs/aspnet/web-api/overview/formats-and-model-binding/json-and-xml-serialization.md

241 lines
12 KiB
Markdown
Raw Normal View History

merge aspnetmigration to master (#2668) * Initial aspnet migration * Updating legacy urls * Clearing .gitignore file Clearing .gitignore file because this isnt a normal .net application where we want to strip away debug files, etc. We want to include everything under these folders. * Updating webhooks toc reference * Removing header debug links * Updating webhooks toc * Updating double quotes in metadata to use single quotes * Moving all code blocks to external files * Updating newlines for step-by-step * Fixing indent problem on some code blocks * Fixing newlines in alt attribute for images; Fixing some missing code block references * Resyncing with live content * Refreshing content from production * Trying to get pdf to generate for our aspnetmigration branch * Update .openpublishing.publish.config.json * Refresh from prod; Removing some legacy urls for pages that shouldnt have it * Updating index pages and removing legacy urls * Updating warning and caution notes * Removing downloads * remove aspnet from exclude list (#2549) * First pass at language detection * Updating author and adding in msc.type * Updating code blocks * Updating note styles * Fixing note styles * Updating docfx.json file to allow pdfs and gifs * Fixing note stylings for bold notes * Updating docfx.json to allow exe, zip and wmv files * Fixing note styles with period; Fixing downloads links * Fixing code blocks for razor cs and vb * Fixing more downloads links; Fixing a few code blocks * Removing   html entity * Fixing some more note stylings * Syncing with prod * Fixing issues with content * Rebuilding toc file * Adding back in files accidentally deleted * Fixing some security notes * Fixing some note styles * Updating unknown code blocks * Updating article * Fixing link * Fixing link * Fixing link * Fixing invalid characters * preliminary toc changes * update toc * fix toc folder with only one link * Fixing extra heading * Fixing articles * Reworking ajax pages * Fixing encoding issues * Updating markup in articles * Fixing space * Fixing spacing issues with links * Fixing note styles * Fixing inline note styles * Fixing missing image * Adding space * Rolling back gitignore file and adding a new one for /aspnet * Fixing some code blocks * Updating code block language * Renaming file * Updating code language * Fixing code blocks * Fixing code blocks * Fixing spaces before 'using' * Fixing cs to js * Changing Note type * Updating broken reference * Replacing headings with bolds under notes/tips, etc * Fixing markdown for pipes * Another attempted to fix pipe characters * Fixing markdown for pipes * remove text about being thread-safe (#2630) * Fixing spacing issue with list * Trying to fix pipe issue * new how to choose doc * move choose doc to core folder * add api ref * fix link * Adding in ms.assetid * Removing \ufeff * fix link * link to mvc intro instead of webhooks * add scenarios * put core first, vertical orientation for scenarios * reorganize toc, make overview work like core version * fix yaml * fix broken links * Adding space * add download link * tweak tables * eliminate images * eliminate images 2 * tweak scenario section headings * add link to core in asp.net overview * Removing gears * Updating table * Updating code block languages * fix urls (#2663) * Removing embedded in-article TOC from top of articles * fix urls (#2666) * fix urls * fix urls * Removing embedded in-article TOC from top of articles * Revert "Removing embedded in-article TOC from top of articles" This reverts commit ff1c3ccdf1cf2d705e0bb040144a10fa130796f6. * Revert "Removing embedded in-article TOC from top of articles" This reverts commit 17c37c726d930ec6854b545bab076dffda486ebe.
2017-02-04 05:40:22 +08:00
---
uid: web-api/overview/formats-and-model-binding/json-and-xml-serialization
merge aspnetmigration to master (#2668) * Initial aspnet migration * Updating legacy urls * Clearing .gitignore file Clearing .gitignore file because this isnt a normal .net application where we want to strip away debug files, etc. We want to include everything under these folders. * Updating webhooks toc reference * Removing header debug links * Updating webhooks toc * Updating double quotes in metadata to use single quotes * Moving all code blocks to external files * Updating newlines for step-by-step * Fixing indent problem on some code blocks * Fixing newlines in alt attribute for images; Fixing some missing code block references * Resyncing with live content * Refreshing content from production * Trying to get pdf to generate for our aspnetmigration branch * Update .openpublishing.publish.config.json * Refresh from prod; Removing some legacy urls for pages that shouldnt have it * Updating index pages and removing legacy urls * Updating warning and caution notes * Removing downloads * remove aspnet from exclude list (#2549) * First pass at language detection * Updating author and adding in msc.type * Updating code blocks * Updating note styles * Fixing note styles * Updating docfx.json file to allow pdfs and gifs * Fixing note stylings for bold notes * Updating docfx.json to allow exe, zip and wmv files * Fixing note styles with period; Fixing downloads links * Fixing code blocks for razor cs and vb * Fixing more downloads links; Fixing a few code blocks * Removing   html entity * Fixing some more note stylings * Syncing with prod * Fixing issues with content * Rebuilding toc file * Adding back in files accidentally deleted * Fixing some security notes * Fixing some note styles * Updating unknown code blocks * Updating article * Fixing link * Fixing link * Fixing link * Fixing invalid characters * preliminary toc changes * update toc * fix toc folder with only one link * Fixing extra heading * Fixing articles * Reworking ajax pages * Fixing encoding issues * Updating markup in articles * Fixing space * Fixing spacing issues with links * Fixing note styles * Fixing inline note styles * Fixing missing image * Adding space * Rolling back gitignore file and adding a new one for /aspnet * Fixing some code blocks * Updating code block language * Renaming file * Updating code language * Fixing code blocks * Fixing code blocks * Fixing spaces before 'using' * Fixing cs to js * Changing Note type * Updating broken reference * Replacing headings with bolds under notes/tips, etc * Fixing markdown for pipes * Another attempted to fix pipe characters * Fixing markdown for pipes * remove text about being thread-safe (#2630) * Fixing spacing issue with list * Trying to fix pipe issue * new how to choose doc * move choose doc to core folder * add api ref * fix link * Adding in ms.assetid * Removing \ufeff * fix link * link to mvc intro instead of webhooks * add scenarios * put core first, vertical orientation for scenarios * reorganize toc, make overview work like core version * fix yaml * fix broken links * Adding space * add download link * tweak tables * eliminate images * eliminate images 2 * tweak scenario section headings * add link to core in asp.net overview * Removing gears * Updating table * Updating code block languages * fix urls (#2663) * Removing embedded in-article TOC from top of articles * fix urls (#2666) * fix urls * fix urls * Removing embedded in-article TOC from top of articles * Revert "Removing embedded in-article TOC from top of articles" This reverts commit ff1c3ccdf1cf2d705e0bb040144a10fa130796f6. * Revert "Removing embedded in-article TOC from top of articles" This reverts commit 17c37c726d930ec6854b545bab076dffda486ebe.
2017-02-04 05:40:22 +08:00
title: "JSON and XML Serialization in ASP.NET Web API | Microsoft Docs"
author: MikeWasson
description: ""
ms.author: aspnetcontent
manager: wpickett
ms.date: 05/30/2012
ms.topic: article
ms.assetid: 1cd7525d-de5e-4ab6-94f0-51480d3255d1
ms.technology: dotnet-webapi
ms.prod: .net-framework
msc.legacyurl: /web-api/overview/formats-and-model-binding/json-and-xml-serialization
msc.type: authoredcontent
---
JSON and XML Serialization in ASP.NET Web API
====================
by [Mike Wasson](https://github.com/MikeWasson)
This article describes the JSON and XML formatters in ASP.NET Web API.
In ASP.NET Web API, a *media-type formatter* is an object that can:
- Read CLR objects from an HTTP message body
- Write CLR objects into an HTTP message body
Web API provides media-type formatters for both JSON and XML. The framework inserts these formatters into the pipeline by default. Clients can request either JSON or XML in the Accept header of the HTTP request.
## Contents
- [JSON Media-Type Formatter](#json_media_type_formatter)
- [Read-Only Properties](#json_readonly)
- [Dates](#json_dates)
- [Indenting](#json_indenting)
- [Camel Casing](#json_camelcasing)
- [Anonymous and Weakly-Typed Objects](#json_anon)
- [XML Media-Type Formatter](#xml_media_type_formatter)
- [Read-Only Properties](#xml_readonly)
- [Dates](#xml_dates)
- [Indenting](#xml_indenting)
- [Setting Per-Type XML Serializers](#xml_pertype)
- [Removing the JSON or XML Formatter](#removing_the_json_or_xml_formatter)
- [Handling Circular Object References](#handling_circular_object_references)
- [Testing Object Serialization](#testing_object_serialization)
<a id="json_media_type_formatter"></a>
## JSON Media-Type Formatter
JSON formatting is provided by the **JsonMediaTypeFormatter** class. By default, **JsonMediaTypeFormatter** uses the [Json.NET](http://json.codeplex.com/) library to perform serialization. Json.NET is a third-party open source project.
If you prefer, you can configure the **JsonMediaTypeFormatter** class to use the **DataContractJsonSerializer** instead of Json.NET. To do so, set the **UseDataContractJsonSerializer** property to **true**:
[!code-csharp[Main](json-and-xml-serialization/samples/sample1.cs)]
### JSON Serialization
This section describes some specific behaviors of the JSON formatter, using the default [Json.NET](http://json.codeplex.com/) serializer. This is not meant to be comprehensive documentation of the Json.NET library; for more information, see the [Json.NET Documentation](http://james.newtonking.com/projects/json/help/).
#### What Gets Serialized?
By default, all public properties and fields are included in the serialized JSON. To omit a property or field, decorate it with the **JsonIgnore** attribute.
[!code-csharp[Main](json-and-xml-serialization/samples/sample2.cs)]
If you prefer an &quot;opt-in&quot; approach, decorate the class with the **DataContract** attribute. If this attribute is present, members are ignored unless they have the **DataMember**. You can also use **DataMember** to serialize private members.
[!code-csharp[Main](json-and-xml-serialization/samples/sample3.cs)]
<a id="json_readonly"></a>
### Read-Only Properties
Read-only properties are serialized by default.
<a id="json_dates"></a>
### Dates
By default, Json.NET writes dates in [ISO 8601](http://www.w3.org/TR/NOTE-datetime) format. Dates in UTC (Coordinated Universal Time) are written with a "Z" suffix. Dates in local time include a time-zone offset. For example:
[!code-console[Main](json-and-xml-serialization/samples/sample4.cmd)]
By default, Json.NET preserves the time zone. You can override this by setting the DateTimeZoneHandling property:
[!code-csharp[Main](json-and-xml-serialization/samples/sample5.cs)]
If you prefer to use [Microsoft JSON date format](https://msdn.microsoft.com/en-us/library/bb299886.aspx#intro_to_json_sidebarb) (`"\/Date(ticks)\/"`) instead of ISO 8601, set the **DateFormatHandling** property on the serializer settings:
[!code-csharp[Main](json-and-xml-serialization/samples/sample6.cs)]
<a id="json_indenting"></a>
### Indenting
To write indented JSON, set the **Formatting** setting to **Formatting.Indented**:
[!code-csharp[Main](json-and-xml-serialization/samples/sample7.cs)]
<a id="json_camelcasing"></a>
### Camel Casing
To write JSON property names with camel casing, without changing your data model, set the **CamelCasePropertyNamesContractResolver** on the serializer:
[!code-csharp[Main](json-and-xml-serialization/samples/sample8.cs)]
<a id="json_anon"></a>
### Anonymous and Weakly-Typed Objects
An action method can return an anonymous object and serialize it to JSON. For example:
[!code-csharp[Main](json-and-xml-serialization/samples/sample9.cs)]
The response message body will contain the following JSON:
[!code-json[Main](json-and-xml-serialization/samples/sample10.json)]
If your web API receives loosely structured JSON objects from clients, you can deserialize the request body to a **Newtonsoft.Json.Linq.JObject** type.
[!code-csharp[Main](json-and-xml-serialization/samples/sample11.cs)]
However, it is usually better to use strongly typed data objects. Then you don't need to parse the data yourself, and you get the benefits of model validation.
The XML serializer does not support anonymous types or **JObject** instances. If you use these features for your JSON data, you should remove the XML formatter from the pipeline, as described later in this article.
<a id="xml_media_type_formatter"></a>
## XML Media-Type Formatter
XML formatting is provided by the **XmlMediaTypeFormatter** class. By default, **XmlMediaTypeFormatter** uses the **DataContractSerializer** class to perform serialization.
If you prefer, you can configure the **XmlMediaTypeFormatter** to use the **XmlSerializer** instead of the **DataContractSerializer**. To do so, set the **UseXmlSerializer** property to **true**:
[!code-csharp[Main](json-and-xml-serialization/samples/sample12.cs)]
The **XmlSerializer** class supports a narrower set of types than **DataContractSerializer**, but gives more control over the resulting XML. Consider using **XmlSerializer** if you need to match an existing XML schema.
### XML Serialization
This section describes some specific behaviors of the XML formatter, using the default **DataContractSerializer**.
By default, the DataContractSerializer behaves as follows:
- All public read/write properties and fields are serialized. To omit a property or field, decorate it with the **IgnoreDataMember** attribute.
- Private and protected members are not serialized.
- Read-only properties are not serialized. (However, the contents of a read-only collection property are serialized.)
- Class and member names are written in the XML exactly as they appear in the class declaration.
- A default XML namespace is used.
If you need more control over the serialization, you can decorate the class with the **DataContract** attribute. When this attribute is present, the class is serialized as follows:
- &quot;Opt in&quot; approach: Properties and fields are not serialized by default. To serialize a property or field, decorate it with the **DataMember** attribute.
- To serialize a private or protected member, decorate it with the **DataMember** attribute.
- Read-only properties are not serialized.
- To change how the class name appears in the XML, set the *Name* parameter in the **DataContract** attribute.
- To change how a member name appears in the XML, set the *Name* parameter in the **DataMember** attribute.
- To change the XML namespace, set the *Namespace* parameter in the **DataContract** class.
<a id="xml_readonly"></a>
### Read-Only Properties
Read-only properties are not serialized. If a read-only property has a backing private field, you can mark the private field with the **DataMember** attribute. This approach requires the **DataContract** attribute on the class.
[!code-csharp[Main](json-and-xml-serialization/samples/sample13.cs)]
<a id="xml_dates"></a>
### Dates
Dates are written in ISO 8601 format. For example, &quot;2012-05-23T20:21:37.9116538Z&quot;.
<a id="xml_indenting"></a>
### Indenting
To write indented XML, set the **Indent** property to **true**:
[!code-csharp[Main](json-and-xml-serialization/samples/sample14.cs)]
<a id="xml_pertype"></a>
## Setting Per-Type XML Serializers
You can set different XML serializers for different CLR types. For example, you might have a particular data object that requires **XmlSerializer** for backward compatibility. You can use **XmlSerializer** for this object and continue to use **DataContractSerializer** for other types.
To set an XML serializer for a particular type, call **SetSerializer**.
[!code-csharp[Main](json-and-xml-serialization/samples/sample15.cs)]
You can specify an **XmlSerializer** or any object that derives from **XmlObjectSerializer**.
<a id="removing_the_json_or_xml_formatter"></a>
## Removing the JSON or XML Formatter
You can remove the JSON formatter or the XML formatter from the list of formatters, if you do not want to use them. The main reasons to do this are:
- To restrict your web API responses to a particular media type. For example, you might decide to support only JSON responses, and remove the XML formatter.
- To replace the default formatter with a custom formatter. For example, you could replace the JSON formatter with your own custom implementation of a JSON formatter.
The following code shows how to remove the default formatters. Call this from your **Application\_Start** method, defined in Global.asax.
[!code-csharp[Main](json-and-xml-serialization/samples/sample16.cs)]
<a id="handling_circular_object_references"></a>
## Handling Circular Object References
By default, the JSON and XML formatters write all objects as values. If two properties refer to the same object, or if the same object appears twice in a collection, the formatter will serialize the object twice. This is a particular problem if your object graph contains cycles, because the serializer will throw an exception when it detects a loop in the graph.
Consider the following object models and controller.
[!code-csharp[Main](json-and-xml-serialization/samples/sample17.cs)]
Invoking this action will cause the formatter to thrown an exception, which translates to a status code 500 (Internal Server Error) response to the client.
To preserve object references in JSON, add the following code to **Application\_Start** method in the Global.asax file:
[!code-csharp[Main](json-and-xml-serialization/samples/sample18.cs)]
Now the controller action will return JSON that looks like this:
[!code-json[Main](json-and-xml-serialization/samples/sample19.json)]
Notice that the serializer adds an &quot;$id&quot; property to both objects. Also, it detects that the Employee.Department property creates a loop, so it replaces the value with an object reference: {&quot;$ref&quot;:&quot;1&quot;}.
> [!NOTE]
> Object references are not standard in JSON. Before using this feature, consider whether your clients will be able to parse the results. It might be better simply to remove cycles from the graph. For example, the link from Employee back to Department is not really needed in this example.
To preserve object references in XML, you have two options. The simpler option is to add `[DataContract(IsReference=true)]` to your model class. The *IsReference* parameter enables object references. Remember that **DataContract** makes serialization opt-in, so you will also need to add **DataMember** attributes to the properties:
[!code-csharp[Main](json-and-xml-serialization/samples/sample20.cs)]
Now the formatter will produce XML similar to following:
[!code-xml[Main](json-and-xml-serialization/samples/sample21.xml)]
If you want to avoid attributes on your model class, there is another option: Create a new type-specific **DataContractSerializer** instance and set *preserveObjectReferences* to **true** in the constructor. Then set this instance as a per-type serializer on the XML media-type formatter. The following code show how to do this:
[!code-csharp[Main](json-and-xml-serialization/samples/sample22.cs?highlight=3)]
<a id="testing_object_serialization"></a>
## Testing Object Serialization
As you design your web API, it is useful to test how your data objects will be serialized. You can do this without creating a controller or invoking a controller action.
2017-09-20 15:36:55 +08:00
[!code-csharp[Main](json-and-xml-serialization/samples/sample23.cs)]