您的位置:首页 > Web前端 > JavaScript

Newtonsoft.Json高级用法

2016-09-08 13:18 330 查看

Newtonsoft.Json介绍

在做开发的时候,很多数据都是以Json格式传输的,而使用Json的时候,我们通常会涉及到几个序列化对象的使用:

System.Runtime.Serialization.Json.DataContractJsonSerializer

System.Web.Script.Serialization.JavaScriptSerializer

Json.NET即Newtonsoft.Json

由于Json.NET的性能及通用性较好,因此它的使用范围越来越广,借助前人的经验今天我也总结使用一下。

官网API:http://www.newtonsoft.com/json/help/html/N_Newtonsoft_Json.htm

基本用法

Json.Net是支持序列化和反序列化DataTable、DataSet、Entity Framework和Entity的

private void Newtonsoft()
{
DataTable dt = new DataTable();
dt.Columns.Add("Name", Type.GetType("System.String"));
dt.Columns.Add("Sex", Type.GetType("System.String"));
dt.Columns.Add("Age", Type.GetType("System.Int32"));
dt.Columns.Add("PhoneNumber", Type.GetType("System.String"));
for (int i = 0; i < 3; i++)
{
DataRow dr = dt.NewRow();
dr["Name"] = "Name" + i;
dr["Sex"] = i % 2 == 0 ? "女" : "男";
dr["Age"] = 18 + i;
dr["PhoneNumber"] = "18233388888";
dt.Rows.Add(dr);
}
/*序列化DataTable*/
string json = JsonConvert.SerializeObject(dt);
/*反序列化DataTable*/
DataTable dtSource = JsonConvert.DeserializeObject<DataTable>(json);
foreach (DataRow item in dtSource.Rows)
{
string name = item["Name"].ToString();
string sex = item["Sex"].ToString();
int age = int.Parse(item["Age"].ToString());
string phoneNumber = item["PhoneNumber"].ToString();
}
}


DataSet、Entity Framework序列化和DataTable一样,这里就不过多介绍了。

序列化之后结果如下:



不创建新类型对json字符串反序列化

可以利用JObject 对象实现通过[]来读取对象中的属性的功能。

private void TestHttpHelper()
{
string url = "http://openapi.tencentyun.com/v3/user/get_info";
string data = "openid=B624064BA065E01CB73F835017FE96FA&" +
"openkey=5F154D7D2751AEDC8527269006F290F70297B7E54667536C&" +
"appid=2&" +
"sig=VrN%2BTn5J%2Fg4IIo0egUdxq6%2B0otk%3D&" +
"pf=qzone&" +
"format=json&" +
"userip=112.90.139.30";
string result = HttpHelper.HttpGet(url + "?" + data);
JObject obj = JsonConvert.DeserializeObject<JObject>(result);
int int1 = Convert.ToInt32(obj["ret"]);
string str1 = obj["msg"].ToString();
}


高级用法

忽略某些属性

默认值的处理

空值的处理

支持非公共成员

日期处理

自定义序列化的字段名称

动态决定属性是否序列化

一、忽略某些属性

实体类中有很多属性,根据实际情况有些属性不需要序列化返回,可以使用该特性。首先介绍Json.Net序列化的模式:OptOut和OptIn

参数解释
OptOut默认值,类中所有公共成员都会被序列化,如果不想被序列化,可以加特性JsonIgnore
OptIn默认情况下,所有成员都不会被序列化,类中的成员只有加特性Json Property才会被序列化,当类的成员很多,但客户端仅仅需要一部分数据时很有用
只有Name和PhoneNumber

[JsonObject(MemberSerialization.OptIn)]
public class Person
{
[JsonProperty]
public string Name { get; set; }
public string Sex { get; set; }
public int Age { get; set; }
public string Address { get; set; }
[JsonProperty]
public string PhoneNumber { get; set; }
}




不要性别、住址、手机号

[JsonObject(MemberSerialization.OptOut)]
public class Person
{
public string Name { get; set; }
[JsonIgnore]
public string Sex { get; set; }
public int Age { get; set; }
[JsonIgnore]
public string Address { get; set; }
[JsonIgnore]
public string PhoneNumber { get; set; }
}




二、默认值处理

序列化时想忽略默认值属性可以通过JsonSerializerSettings.DefaultValueHanding来设置,该值为枚举值

参数解释
DefaultValueHanding.Ignore序列化和反序列化时,忽略没赋值的属性
DefaultValueHanding.Include序列化和反序列化时,包含没赋值的属性
public class Person
{
public string Name { get; set; }
public string Sex { get; set; }
public int Age { get; set; }
public string Address { get; set; }
public string PhoneNumber { get; set; }
}


忽略没有赋值的属性:



包含没有赋值的属性:



三、空值的处理

序列化时需要忽略值为NULL的属性,可以通过JsonSerializerSettings.NullValueHanding来设置,另外用JsonSerializerSettings设置属性是对序列化过程中所有属性生效的,想单独对某一个属性生效可以使用JsonProperty,下面将展示两种方式。

1、JsonSerializerSettings方式

忽略值为NULL的属性:

int类型属性默认值为0,string类型属性默认值为null



包含值为NULL的属性:



2、JsonProperty方式

当Address值为NULL时忽略:

public class Person
{
public string Name { get; set; }
public string Sex { get; set; }
public int Age { get; set; }
[JsonProperty(NullValueHandling=NullValueHandling.Ignore)]
public string Address { get; set; }
public string PhoneNumber { get; set; }
}




四、支持非公共成员

序列化时默认都是只处理公共成员,如果需要处理非公共成员,就要在该成员上加特性”JsonProperty”。

private不加JsonProperty:

public class Person
{
public string Name { get; set; }
public string Sex { get; set; }
public int Age { get; set; }
public string Address { get; set; }
public string PhoneNumber { get; set; }
private int Wife { get; set; }
public Person()
{
Wife = 3;
}
}




private加JsonProperty:

public class Person
{
public string Name { get; set; }
public string Sex { get; set; }
public int Age { get; set; }
public string Address { get; set; }
public string PhoneNumber { get; set; }
[JsonProperty]
private int Wife { get; set; }
public Person()
{
Wife = 3;
}
}




五、日期处理

DateTime类型属性的处理稍微麻烦点,系统默认格式化为iso日期标准,

public class Person
{
public string Name { get; set; }
public string Sex { get; set; }
public DateTime Birthday { get; set; }
}




public class Person
{
public string Name { get; set; }
public string Sex { get; set; }
[JsonConverter(typeof(IsoDateTimeConverter))]
public DateTime Birthday { get; set; }
}


但实际使用过程中大多数使用的可能是yyyy-MM-dd或yyyy-MM-dd HH:mm:ss这两种格式的日期,解决办法可以是将DateTime类型转成string类型自己先格式化然后再序列化。其次,Json.Net提供了IsoDateTimeConvert日期转换这个类,可以通过JsonConverter实现相应的日期转换,效果跟不加JsonConverter是一样的,在此就不截图了,但这明显不是我们想要的结果,我们可以继承该类实现自己的日期格式

public class Person
{
public string Name { get; set; }
public string Sex { get; set; }
[JsonConverter(typeof(ChinaDateTimeConcerter))]
public DateTime Birthday { get; set; }
}
public class ChinaDateTimeConcerter : DateTimeConverterBase
{
private static IsoDateTimeConverter dtConverter = new IsoDateTimeConverter { DateTimeFormat = "yyyy-MM-dd" };
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
return dtConverter.ReadJson(reader, objectType, existingValue, serializer);
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
dtConverter.WriteJson(writer, value, serializer);
}
}




这样自己就实现了一个yyyy-MM-dd日期格式化转换类,如果需要yyyy-MM-dd HH:mm:ss日期格式自己也可以很容易实现了。

六、自定义序列化字段的名称

实体类中定义的属性名可能不是自己想要的名称,但是不能更改实体定义,这个时候可以自定义序列化字段的名称。

public class Person
{
public string Name { get; set; }
[JsonProperty(PropertyName = "MySex")]
public string Sex { get; set; }
[JsonProperty(PropertyName = "MyAge")]
public int Age { get; set; }
}


序列化后:



反序列化后:



七、动态决定属性是否序列化

在某些场景下,可能A场景输出A、B、C三个属性,B场景输出C,D,E三个属性,Json.Net支持此种非常规的特性。

继承默认的DefaultContractResolver类,传入需要的属性:

public class Person
{
public string Name { get; set; }
public string Sex { get; set; }
public int Age { get; set; }
public string Address { get; set; }
public string PhoneNumber { get; set; }
}
public class LimitPropsContractResolver : DefaultContractResolver
{
string[] props = null;
public LimitPropsContractResolver(string[] props)
{
//指定要序列化属性的清单
this.props = props;
}
protected override IList<JsonProperty> CreateProperties(Type type, MemberSerialization memberSerialization)
{
IList<JsonProperty> list = base.CreateProperties(type, memberSerialization);
//只保留清单列出的属性
return list.Where(p => props.Contains(p.PropertyName)).ToList();
}
}


使用自定义解析类,只输出Name、Sex、Age三个属性



总结

Newtonsoft.Json序列化库替我们想了很多特性,也实现了很多特性,除了上面介绍的几种高级用法外,还有其它的特殊用法,可以去官网进行学习,我在学会了使用Newtonsoft.Json之后,非常想知道其内部是如何实现特性这一功能的,不知道大家有何感想。

官网示例代码地址:http://www.newtonsoft.com/json/help/html/DefaultValueHandlingIgnore.htm
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息