加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

Newtonsoft.Json高级用法

发布时间:2020-12-16 19:34:07 所属栏目:百科 来源:网络整理
导读:Newtonsoft.Json介绍 在做开发的时候,很多数据都是以Json格式传输的,而使用Json的时候,我们通常会涉及到几个序列化对象的使用: System.Runtime.Serialization.Json.DataContractJsonSerializer System.Web.Script.Serialization.JavaScriptSerializer Js

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一样,这里就不过多介绍了。
序列化之后结果如下:

高级用法

  1. 忽略某些属性
  2. 默认值的处理
  3. 空值的处理
  4. 支持非公共成员
  5. 日期处理
  6. 自定义序列化的字段名称
  7. 动态决定属性是否序列化

一、忽略某些属性
实体类中有很多属性,根据实际情况有些属性不需要序列化返回,可以使用该特性。首先介绍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

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读