Json   发布时间:2022-04-22  发布网站:大佬教程  code.js-code.com
大佬教程收集整理的这篇文章主要介绍了Newtonsoft.Json高级用法大佬教程大佬觉得挺不错的,现在分享给大家,也给大家做个参考。


  手机端应用讲究速度快,体验好。刚好手头上的一个项目服务端接口有性能问题,需要进行优化。在接口多次修改中,实体添加了很多字段用于中间计算或者存储,然后最终用Newtonsoft.Json进行序列化返回数据,经过分析一个简单的列表接口每一行数据返回了16个字段,但是手机APP端只用到了其中7个字段,剩余9个字段的数据全部都是多余的,如果接口返回数据为40K大小,也就是说大约20K的数据为无效数据,3G网络下20K下载差不多需要1s,不返回无效数据至少可以节约1s的时间,大大提高用户体验。本篇将为大家介绍Newtonsoft.Json的一些高级用法,可以修改很少的代码解决上述问题。

阅读目录

Newtonsoft.Json介绍

  在做开发的时候,很多数据交换都是以json格式传输的。而使用Json的时候,我们很多时候会涉及到几个序列化对象的使用:DataContractJsonserializer,JavaScriptserializer Json.NET即Newtonsoft.Json。大多数人都会选择性能以及通用性较好Json.NET,这个不是微软的类库,但是一个开源的世界级的Json操作类库,从下面的性能对比就可以看到它的其中之一的性能优点。

齐全的API介绍,使用方式简单

基本用法

  Json.Net是支持序列化和反序列化DataTable,DataSet,Entity Framework和Entity的。下面分别举例说明序列化和反序列化。

DataTable:

            //序列化DataTable
            DataTable dt = new DataTable();
            dt.columns.Add("Age",Type.GetType("system.int32"));
            dt.columns.Add("Name",Type.GetType("System.String"));
            dt.columns.Add("Sex",Type.GetType("System.String"));
            dt.columns.Add("Ismarry",Type.GetType("System.Boolean"));
            for (int i = 0; i < 4; i++)
            {
                Da@R_611_2301@w dr = dt.NewRow();
                dr["Age"] = i + 1;
                dr["Name"] = "Name" + i;
                dr["Sex"] = i % 2 == 0 ? "" : "";
                dr["Ismarry"] = i % 2 > 0 ? true : false;
                dt.Rows.Add(dr);
            }
            Console.WriteLine(JsonConvert.serializeObject(dt));

利用上面字符串进行反序列化

 String json = JsonConvert.serializeObject(dt);
 dt=JsonConvert.DeserializeObject<DataTable>(json);
 foreach (Da@R_611_2301@w dr in dt.Rows)
 {
   Console.WriteLine("{0}\t{1}\t{2}\t{3}\t",dr[0],dr[1],dr[2],dr[3]);
 }

Entity序列化和DataTable一样,就不过多介绍了。

高级用法

1.忽略某些属性

2.值的处理

3.空值的处理

4.支持非公共成员

5.日期处理

6.自定义序列化的字段名称

  7.动态决定属性是否序列化

一.忽略某些属性

  类似本问开头介绍的接口优化,实体中有些属性不需要序列化返回,可以使用该特性。首先介绍Json.Net序列化的模式:OptOut 和 OpTin

OptOut 认值,类中所有公有成员会被序列化,如果不想被序列化,可以用特性JsonIgnore
OpTin 认情况下,所有的成员不会被序列化,类中的成员只有标有特性JsonProperty的才会被序列化,当类的成员很多,但客户端仅仅需要一部分数据时,很有用

 仅需要姓名属性

    [JsonObject(Memberserialization.opTin)]
    public class Person
    {
        public int Age { get; set; }

        [JsonProperty]
        public String Name { get; set; }

        public String Sex { get; set; }

        public bool Ismarry { get; set; }

        public datetiR_162_11845@e Birthday { get; set; }
    }

不需要是否结婚属性

    [JsonObject(Memberserialization.optOut)]
    public class Person
    {
        public int Age { get; set; }

        public String Name { get; set; }

        public String Sex { get; set; }

        [JsonIgnore]
        public bool Ismarry { get; set; }

        public datetiR_162_11845@e Birthday { get; set; }
    }

通过上面的例子可以看到,要实现不返回某些属性的需求很简单。1.在实体类加上[JsonObject(Memberserialization.optOut)] 2.在不需要返回的属性加上 [JsonIgnore]说明。

二.认值处理

序列化时想忽略认值属性可以通过JsonserializerSetTings.DefaultValueHandling来确定,该值为枚举值

DefaultValueHandling.Ignore 
序列化和反序列化时,忽略认值
DefaultValueHandling.Include 
序列化和反序列化时,包含认值
 [DefaultValue(10)]
 public int Age { get; set; }
 Person p = new Person { Age = 10,Name = "张三",Sex = "",Ismarry = false,Birthday = new datetiR_162_11845@e(1991,1,2) };
 JsonserializerSetTings jsetTing=new JsonserializerSetTings();
 jsetTing.DefaultValueHandling=DefaultValueHandling.Ignore;
 Console.WriteLine(JsonConvert.serializeObject(p,FormatTing.Indented,jsetTing));

最终结果如下:

三.空值的处理

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

  1.JsonserializerSetTings

 Person p = new Person { room=null,Age = 10,2) };
 JsonserializerSetTings jsetTing=new JsonserializerSetTings();
 jsetTing.NullValueHandling = NullValueHandling.Ignore;
 Console.WriteLine(JsonConvert.serializeObject(p,jsetTing));

2.JsonProperty

通过JsonProperty属性设置的方法,可以实现某一属性特别处理的需求,如认值处理,空值处理,自定义属性名处理,格式化处理。上面空值处理实现

 [JsonProperty(NullValueHandling=NullValueHandling.IgnorE)]
 public Room room { get; set; }

四.支持非公共成员

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

 [JsonProperty]
 private int Height { get; set; }

五.日期处理

  对于Dateime类型日期的格式化就比较麻烦了,系统自带的会格式化成iso日期标准,但是实际使用过程中大多数使用的可能是yyyy-MM-dd 或者yyyy-MM-dd HH:mm:ss两种格式的日期,解决办法是可以将datetiR_162_11845@e类型改成String类型自己格式化好,然后在序列化。如果不想修改代码,可以采用下面方案实现。

Json.Net提供了IsodatetiR_162_11845@eConverter日期转换这个类,可以通过JsnConverter实现相应的日期转换

    [JsonConverter(typeof(IsodatetiR_162_11845@eConverter))]
    public datetiR_162_11845@e Birthday { get; set; }

  但是IsodatetiR_162_11845@eConverter日期格式不是我们想要的,我们可以继承该类实现自己的日期

    public class ChinadatetiR_162_11845@eConverter : datetiR_162_11845@eConverterBase
    {
        private static IsodatetiR_162_11845@eConverter dtConverter = new IsodatetiR_162_11845@eConverter { datetiR_162_11845@eFormat = "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格式化转换类,可以看到只是初始化IsodatetiR_162_11845@eConverter时给的日期格式为yyyy-MM-dd即可,下面看下效果

[JsonConverter(typeof(ChinadatetiR_162_11845@eConverter))]
public datetiR_162_11845@e Birthday { get; set; }

可以根据自己需求实现不同的转换类

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

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

     [JsonProperty(PropertyName = "CName")]
     public String Name { get; set; }

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

  这个是为了实现@米粒儿提的需求特别增加的,根据某些场景,可能A场景输出A,B,C三个属性,B场景输出E,F属性然实际中不一定存在这种需求,但是json.net依然可以支持该特性。

  继承认的DefaultContractResolver类,传入需要输出属性

public class LimitPropsContractResolver : DefaultContractResolver
    {
        String[] props = null;

        public LimitPropsContractResolver(String[] props)
        {
            //指定要序列化属性的清单
            this.props = props;
        }

        //REF: http://james.newtonking.com/archive/2009/10/23/efficient-json-with-json-net-reducing-serialized-json-size.aspx

        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();
        }
    }
        public int Age { get; set; }

        [JsonIgnore]
        public bool Ismarry { get; set; }

        public String Sex { get; set; }
      JsonserializerSetTings jsetTing=new JsonserializerSetTings();
      jsetTing.ContractResolver = new LimitPropsContractResolver(new String[] { "Age","Ismarry" });
      Console.WriteLine(JsonConvert.serializeObject(p,jsetTing));

使用自定义的解析类,只输出"Age","Ismarry"两个属性,看下最终结果.只输出了Age属性,为什么Ismarry属性没有输出呢,因为标注了JsonIgnore

看到上面的结果想要实现pc端序列化一部分,手机端序列化另一部分就很简单了吧,我们改下代码实现一下

  String[] propNames = null;
  if (p.Age > 10)
  {
    propNames = new String[] { "Age","Ismarry" };
  }
  else
  {
      propNames = new String[] { "Age","Sex" };
  }
  jsetTing.ContractResolver = new LimitPropsContractResolver(propNames);
  Console.WriteLine(JsonConvert.serializeObject(p,jsetTing));

大佬总结

以上是大佬教程为你收集整理的Newtonsoft.Json高级用法全部内容,希望文章能够帮你解决Newtonsoft.Json高级用法所遇到的程序开发问题。

如果觉得大佬教程网站内容还不错,欢迎将大佬教程推荐给程序员好友。

本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。