A light weight JSON serialization library written in C#.
Supported .NET platform:
- .NET Framework 3.5
- .NET Framework 4.x
- All other platforms that support .NET Standard 2, such as .NET Core 2/3, .NET 5/6
Dependency:
- cmstar.RapidReflection To emit IL for accessing type members.
Install via Package Manager:
Install-Package cmstar.Serialization.Json
or via dotnet-cli:
dotnet add package cmstar.Serialization.Json
The JsonSerializer
class is the entry for serializing/deserializing.
public class Data
{
public string String { get; set; }
public int Int { get; set; }
public int[] Array { get; set; }
}
var serializer = new JsonSerializer();
serializer.Serialize(123);
//-> 123
serializer.Serialize("Hello\nWorld");
//-> "Hellow\nWorld"
serializer.Serialize(DateTime.Now.ToUniversalTime());
//-> "2013-07-15T14:21:05.2151663Z"
serializer.Serialize(new char[] { 'a', 'b' });
//-> ["a","b"]
serializer.Serialize(new Dictionary<string, int> {
{ "key1", 1 },
{ "key2", 2 }
});
//-> {"key1":1,"key2":2}
serializer.Serialize(new Data { Array = new int[] { 1, 2 } });
//-> {"String":null,"Int":0,"Array":[1,2]}
// non-generic version
Data data = (Data)serializer.Deserialize(
"{\"String\":null,\"Int\":0,\"Array\":[1,2]}",
typeof(Data));
// generic version
int[] array = serializer.Deserialize<int[]>("[1,2,3]");
Serializing anonymouse objects is just the same:
var anonymousObject = new {
Foo = 123,
Bar = "xx",
Array = new int[] { 1, 2, 3 }
};
JsonSerializer.Default.Serialize(anonymousObject);
//-> {"Foo":123,"Bar":"xx","Array":[1,2,3]}
To deserialize, a template object should be provided:
var template = new { Foo = 0, Bar = (string)null };
var json = "{\"Foo\":10,\"Bar\":\"s\"}";
// call JsonSerializer.Deserialize<T>(string json, T template)
var result = JsonSerializer.Default.Deserialize(json, template);
Each instance of JsonSerializer
is isolated, it can keep different instances of JsonContract
and can be customized separately from another JsonSerializer
. But in most time, we need just one JsonSerializer
, in the case we can use JsonSerializer.Default
:
var json = JsonSerializer.Default.Serialize(new Data());
var data = JsonSerializer.Default.Deserialize<Data>(json);
To serialize a POCO, by default, only public properties will be serialized.
You can use the JsonPropertyAttribute
to select the members you need:
class Data
{
public Data(string s) { String = s; }
[JsonProperty("string_value")] // mark a private field
private string String;
[JsonProperty] // no name specified, will use 'Int' directly
public int Int { get; set; }
public int WillBeIngored { get; set; }
}
JsonSerializer.Default.Serialize(new Data("s") { Int = 3 });
//-> {"string_value":"s","Int":3}
or use JsonIgnoreAttribute
:
class Data
{
public string String { get; set; }
[JsonJsonIgnore]
public int Int { get; set; }
}
JsonSerializer.Default.Serialize(new Data { String = "s", Int = 3 });
//-> {"String":"s"}
Note: If you mix
JsonIgnoreAttribute
andJsonPropertyAttribute
together, the serializer ignoresJsonPropertyAttribute
.
Note: If a property has no getter accessor (public or non-public), it will be ignored during the serialization; and the value of a property without a setter accessor will not be set.
By default the JSONs outputted is compact but not much human-readable. An overload of the JsonSerializer.Serialize()
mothod accepts an argument 'formatting' which can be used to specify the format of JSON seriliazed.
JsonSerializer.Default.Serialize(new Data { Array = new int[] { 1, 2 } });
//-> {"String":null,"Int":0,"Array":[1,2]}
JsonSerializer.Default.Serialize(
new Data { Array = new int[] { 1, 2 } },
Formatting.Multiple);
/* ->
{
"String":null,
"Int":0,
"Array":[
1,
2
]
}
*/
JsonSerializer.Default.Serialize(
new Data { Array = new int[] { 1, 2 } },
Formatting.Indented);
/* ->
{
"String":null,
"Int":0,
"Array":[
1,
2
]
}
*/
For performance need, the JsonSerializer.FastSerialize()
method provides a faster serialization, which is about 50% faster than the JsonSerializer.Serialize()
method.
The faster version uses the JsonWriter
class against the JsonWriterImproved
class used by the JsonSerializer.Serialize()
method. See the 'JsonWriter' section below for more details.
The classes derive from the JsonContract
class indicate how to serialize CLR objects or deserialize JSONs.
The table below gives out the default contracts, which will convert the CLR types to/from corresponding JSON types:
CLR type | Contract | JSON type |
---|---|---|
String | StringContract | String |
Char | StringContract | String |
Boolean | BooleanContract | Boolean |
SByte | NumberContarct | Number |
Int16 | NumberContarct | Number |
Int32 | NumberContarct | Number |
Int64 | NumberContarct | Number |
Byte | NumberContarct | Number |
UInt16 | NumberContarct | Number |
UInt32 | NumberContarct | Number |
UInt64 | NumberContarct | Number |
IntPtr | NumberContarct | Number |
UIntPtr | NumberContarct | Number |
Single | NumberContarct | Number |
Double | NumberContarct | Number |
Decimal | NumberContarct | Number |
DateTimeOffset | DateTimeOffsetContarct | String |
DateTime | DateTimeContarct | String |
Guid | GuidContarct | String |
Nullable<T> | NullableTypeContract | Depends on typeof(T) |
Types derived from Enum | EnumContarct | Number |
Implementations of IDictionary | DictionaryContarct | Object |
Implementations of IDictionary<K,V> | DictionaryContarct | Object |
Implementations of ICollection | ArrayContarct | Array |
Implementations of ICollection<T> | ArrayContarct | Array |
Other types not listed above | ObjectContarct | Object |
- CLR
null
(Nothing
in VB.net) will be serialized to JSONnull
. - For an object of type
Nullable<T>
, if has value, it will be serialized using the underlying value; otherwise, will be serialized to JSONnull
. - Non-generic implementations of
ICollection
orIDictionary
can be serialized but can not be deserialized because when deserializing the application doesn't know which CLR type should be used - the JSON type to CLR type mapping is 1 to N.
This class is used to resolve the JsonContract
s for different types.
You can register custom JsonContract
s by sending a dictionary to the constructor of JsonContractResolver
:
var customContracts = new Dictionary<Type, JsonContract>();
customContracts.Add(typeof(Data), new CustomDataContract());
var contractResolver = new JsonContractResolver(customContracts);
var serializer = new JsonSerializer(contractResolver);
Note: You can't register custom JsonContract
s to JsonSerializer.Default
at present.
The default contract for DateTime
/DateTimeOffset
is the DateTimeContract
/DateTimeOffsetContract
,
which will serialize dates in the ISO-8601 format yyyy-MM-ddTHH:mm:ss.ffffffZ
,
such as 2022-01-31T13:15:05.2151663-02:00
, or 2022-01-31T13:15:05.2151663Z
(UTC).
You can register CustomFormatDateTimeOffsetContract
to customize the format, with a property Format
,
the code below shows how to serialize dates in the format yyyy~MM~dd HH:mm:ss
:
DateTimeContract
shares the format of DateTimeOffsetContract
,
Change the format for DateTimeOffset
will also change the format for DateTime
.
var dateTimeContract = new CustomFormatDateTimeOffsetContract();
dateTimeContract.Format = "yyyy~MM~dd HH@mm@ss";
var customContracts = new Dictionary<Type, JsonContract>();
customContracts.Add(typeof(DateTimeOffset), dateTimeContract);
var contractResolver = new JsonContractResolver(customContracts);
var serializer = new JsonSerializer(contractResolver);
serializer.Serialize(DateTimeOffset.Now);
//-> "2013~07~15 14@41@03"
// When serializing DateTime, it shares the format.
serializer.Serialize(DateTime.Now);
//-> "2013~07~15 14@41@03"
Another contract provided is MicrosoftJsonDateContract
, which formats time
in the Miscrosoft JSON format, such as /Date(1620142251000+0300)/
.
By default enums are serialized to JSON numbers using the index, if you need the name of an enum, you can setup the EnumContract.UseEnumName
property to true
.
var stringEnumContract = new EnumContract(typeof(SomEnum));
stringEnumContract.UseEnumName = true;
var customContracts = new Dictionary<Type, JsonContract>();
customContracts.Add(typeof(SomEnum), stringEnumContract);
var contractResolver = new JsonContractResolver(customContracts);
var serializer = new JsonSerializer(contractResolver);
serializer.Serialize(SomEnum.SomeItem);
//-> "SomeItem"
Here is an example that shows how to tell the JsonSerializer
to serialize all enums by their names.
First, build a sub class of the JsonContractResolver
and override the DoResove
method which is the core method for contract resolving:
class StringEnumContractResolver : JsonContractResolver
{
protected override JsonContract DoResolve(Type type)
{
if (type.IsSubclassOf(typeof(Enum)))
return new EnumContract(type) { UseEnumName = true };
return base.DoResolve(type);
}
}
Then you can setup the JsonSerializer
with the class above:
var contractResolver = new StringEnumContractResolver();
var serializer = new JsonSerializer(contractResolver);
serializer.Serialize(SomeEnum.SomeItem);
//-> "SomeItem"