Newtonsoft.Json使用技巧

来源:https://www.cnblogs.com/xtxk110/archive/2019/10/11/11654486.html
-Advertisement-
Play Games

本篇將為大家介紹Newtonsoft.Json的一些高級用法,可以修改很少的代碼解決上述問題。 閱讀目錄 Newtonsoft.Json介紹 基本用法 高級用法 總結 回到頂部 Newtonsoft.Json介紹 在做開發的時候,很多數據交換都是以json格式傳輸的。而使用Json的時候,我們很多時 ...


  本篇將為大家介紹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++)
            {
                DataRow 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 (DataRow 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.動態決定屬性是否序列化

    8.枚舉值的自定義格式化問題

  9.自定義類型轉換

  10.全局序列化設置

 一.忽略某些屬性

  類似本問開頭介紹的介面優化,實體中有些屬性不需要序列化返回,可以使用該特性。首先介紹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 DateTime 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 DateTime 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 DateTime(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, Name = "張三豐", Sex = "", IsMarry = false, Birthday = new DateTime(1991, 1, 2) };
 JsonSerializerSettings jsetting=new JsonSerializerSettings();
 jsetting.NullValueHandling = NullValueHandling.Ignore;
 Console.WriteLine(JsonConvert.SerializeObject(p, Formatting.Indented, 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兩種格式的日期,解決辦法是可以將DateTime類型改成string類型自己格式化好,然後在序列化。如果不想修改代碼,可以採用下麵方案實現。

      Json.Net提供了IsoDateTimeConverter日期轉換這個類,可以通過JsnConverter實現相應的日期轉換

    [JsonConverter(typeof(IsoDateTimeConverter))]
    public DateTime Birthday { get; set; }

  但是IsoDateTimeConverter日期格式不是我們想要的,我們可以繼承該類實現自己的日期

複製代碼
    public class ChinaDateTimeConverter : 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格式化轉換類,可以看到只是初始化IsoDateTimeConverter時給的日期格式為yyyy-MM-dd即可,下麵看下效果

[JsonConverter(typeof(ChinaDateTimeConverter))]
public DateTime Birthday { get; set; }

   可以根據自己需求實現不同的轉換類

 

六.自定義序列化的欄位名稱

    實體中定義的屬性名可能不是自己想要的名稱,但是又不能更改實體定義,這個時候可以自定義序列化欄位名稱。

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

 

七.動態決定屬性是否序列化

  這個是為了實現@米粒兒提的需求特別增加的,根據某些場景,可能A場景輸出A,B,C三個屬性,B場景輸出E,F屬性。雖然實際中不一定存在這種需求,但是json.net依然可以支持該特性。

  繼承預設的DefaultContractResolver類,傳入需要輸出的屬性

     重寫修改了一下,大多數情況下應該是要排除的欄位少於要保留的欄位,  為了方便書寫這裡修改了構造函數加入retain表示props是需要保留的欄位還是要排除的欄位

複製代碼
public class LimitPropsContractResolver : DefaultContractResolver
    {
        string[] props = null;

        bool retain;

        /// <summary>
        /// 構造函數
        /// </summary>
        /// <param name="props">傳入的屬性數組</param>
        /// <param name="retain">true:表示props是需要保留的欄位  false:表示props是要排除的欄位</param>
        public LimitPropsContractResolver(string[] props, bool retain=true)
        {
            //指定要序列化屬性的清單
            this.props = props;

            this.retain = retain;
        }

        protected override IList<JsonProperty> CreateProperties(Type type,

        MemberSerialization memberSerialization)
        {
            IList<JsonProperty> list =
            base.CreateProperties(type, memberSerialization);
            //只保留清單有列出的屬性
            return list.Where(p => {
                if (retain)
                {
                    return props.Contains(p.PropertyName);
                }
                else
                {
                    return !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, Formatting.Indented, 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, Formatting.Indented, jsetting));
複製代碼

 

八.枚舉值的自定義格式化問題

  預設情況下對於實體裡面的枚舉類型系統是格式化成改枚舉對應的整型數值,那如果需要格式化成枚舉對應的字元怎麼處理呢?Newtonsoft.Json也幫我們想到了這點,下麵看實例

複製代碼
    public enum NotifyType
    {
        /// <summary>
        /// Emil發送
        /// </summary>
        Mail=0,

        /// <summary>
        /// 簡訊發送
        /// </summary>
        SMS=1
    }

    public class TestEnmu
    {
        /// <summary>
        /// 消息發送類型
        /// </summary>
        public NotifyType Type { get; set; }
    }
    JsonConvert.SerializeObject(new TestEnmu());
複製代碼

輸出結果:  現在改造一下,輸出"Type":"Mail"

複製代碼
    public class TestEnmu
    {
        /// <summary>
        /// 消息發送類型
        /// </summary>
        [JsonConverter(typeof(StringEnumConverter))]
        public NotifyType Type { get; set; }
    }
複製代碼

其它的都不變,在Type屬性上加上了JsonConverter(typeof(StringEnumConverter))表示將枚舉值轉換成對應的字元串,而StringEnumConverter是Newtonsoft.Json內置的轉換類型,最終輸出結果

 

九.自定義類型轉換

預設情況下對於實體裡面的Boolean系統是格式化成true或者false,對於true轉成"是" false轉成"否"這種需求改怎麼實現了?我們可以自定義類型轉換實現該需求,下麵看實例

複製代碼
public class BoolConvert : JsonConverter
    {
        private string[] arrBString { get; set; }

        public BoolConvert()
        {
            arrBString = "是,否".Split(',');
        }

        /// <summary>
        /// 構造函數
        /// </summary>
        /// <param name="BooleanString">將bool值轉換成的字元串值</param>
        public BoolConvert(string BooleanString)
        {
            if (string.IsNullOrEmpty(BooleanString))
            {
                throw new ArgumentNullException();
            }
            arrBString = BooleanString.Split(',');
            if (arrBString.Length != 2)
            {
                throw new ArgumentException("BooleanString格式不符合規定");
            }
        }


        public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
        {
            bool isNullable = IsNullableType(objectType);
            Type t = isNullable ? Nullable.GetUnderlyingType(objectType) : objectType;

            if (reader.TokenType == JsonToken.Null)
            {
                if (!IsNullableType(objectType))
                {
                    throw new Exception(string.Format("不能轉換null value to {0}.", objectType));
                }

                return null;
            }

            try
            {
                if (reader.TokenType == JsonToken.String)
                {
                    string boolText = reader.Value.ToString();
                    if (boolText.Equals(arrBString[0], StringComparison.OrdinalIgnoreCase))
                    {
                        return true;
                    }
                    else if (boolText.Equals(arrBString[1], StringComparison.OrdinalIgnoreCase))
                    {
                        return false;
                    }
                }

                if (reader.TokenType == JsonToken.Integer)
                {
                    //數值
                    return Convert.ToInt32(reader.Value) == 1;
                }
            }
            catch (Exception ex)
            {
                throw new Exception(string.Format("Error converting value {0} to type '{1}'", reader.Value, objectType));
            }
            throw new Exception(string.Format("Unexpected token {0} when parsing enum", reader.TokenType));
        }

        /// <summary>
        /// 判斷是否為Bool類型
        /// </summary>
        /// <param name="objectType">類型</param>
        /// <returns>為bool類型則可以進行轉換</returns>
        public override bool CanConvert(Type objectType)
        {
            return true;
        }


        public bool IsNullableType(Type t)
        {
            if (t == null)
            {
                throw new ArgumentNullException("t");
            }
            return (t.BaseType.FullName=="System.ValueType" && t.GetGenericTypeDefinition() == typeof(Nullable<>));
        }

        public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
        {
            if (value == null)
            {
                writer.WriteNull();
                return;
            }

            bool bValue = (bool)value;

            if (bValue)
            {
                writer.WriteValue(arrBString[0]);
            }
            else
            {
                writer.WriteValue(arrBString[1]);
            }
        }
    }
複製代碼

 自定義了BoolConvert類型,繼承自JsonConverter。構造函數參數BooleanString可以讓我們自定義將true false轉換成相應字元串。下麵看實體裡面怎麼使用這個自定義轉換類型

    public class Person
    {
        [JsonConverter(typeof(BoolConvert))]
        public bool IsMarry { get; set; }
    }

相應的有什麼個性化的轉換需求,都可以使用自定義轉換類型的方式實現。

 

十.全局序列化設置

文章開頭提出了Null值欄位怎麼不返回的問題,相應的在高級用法也給出了相應的解決方案使用jsetting.NullValueHandling = NullValueHandling.Ignore; 來設置不返回空值。這樣有個麻煩的地方,每個不想返回空值的序列化都需設置一下。可以對序列化設置一些預設值方式麽?下麵將解答

複製代碼
  Newtonsoft.Json.JsonSerializerSettings setting = new Newtonsoft.Json.JsonSerializerSettings();
   JsonConvert.DefaultSettings = new Func<JsonSerializerSettings>(() =>
   {
    //日期類型預設格式化處理   setting.DateFormatHandling
= Newtonsoft.Json.DateFormatHandling.MicrosoftDateFormat; setting.DateFormatString = "yyyy-MM-dd HH:mm:ss";
    //空值處理 setting.NullValueHandling
= NullValueHandling.Ignore;

//高級用法九中的Bool類型轉換 設置 setting.Converters.Add(
new BoolConvert("是,否")); return setting; });
複製代碼

 

這樣設置以後,以後使用序列化的地方就不需要單獨設置了,個人最喜歡設置的是空值處理這一塊。


總結

  Newtonsoft.Json序列化庫替我們想了很多特性,也實現了很多特性,除了上面介紹的幾種高級用法外,還有其它的特殊用法,可以去官網進行學習。當然這裡我目前最喜歡的特性就是那個忽略部分屬性序列化的功能,很小的代碼改動實現了介面的優化,提升了用戶體驗。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 特殊語句 yield語句 運行結果: checked 和 unchecked 語句 用於控制整型類型算術運算和轉換的溢出檢查上下文 lock語句 它的作用是鎖定某一代碼塊,讓同一時間只有一個線程訪問該代碼塊 ...
  • asp.net core 在iis上運行拋出502.5運行拋出異常 環境說明 已安裝 .net core runtime 2.1.401 已安裝 .net core 2.1.3 運行狀況說明: 直接在命令行運行dotnet 命令可以啟動站點 訪問iis的站點拋出502.5錯誤 可能原因已經解決方法 ...
  • public class MD5Help { //此代碼示例通過創建哈希字元串適用於任何 MD5 哈希函數 (在任何平臺) 上創建 32 個字元的十六進位格式哈希字元串官網案例改編 /// <summary> ///獲取32位md5加密 /// </summary> /// <param name=... ...
  • 首先,我們先來實現主界面的扁平化 此處分為兩個步驟,第一步是更改winform自帶的MainForm窗體屬性,第二步是添加窗體事件。 將主窗體FormBorderStyle更改為None,這樣就得到了一個無邊框的窗體(winform自帶邊框太醜。。) 調節背景色,建議找到自己喜歡的顏色,然後使用取色 ...
  • 為表示對原作者的尊重,先貼出原文鏈接:https://www.cnblogs.com/hnsongbiao/p/8250956.html 以下是正文: dynamic類型簡單示例 動態添加欄位 List<string> fieldList = new List<string>() { "Name", ...
  • 2019 中國.NET 開發者峰會正式啟動 目 錄 1. 概述... 2 2. 準備運行程式包... 2 3. 安裝.netcore. 3 4. 安裝mysql 3 5. 安裝nginx. 5 6. 配置iNeuOS資料庫和文件... 5 7. 運行iNeuOS. 6 1. 概述 參見前兩篇文章:《 ...
  • Wpf file embeded resource will compile the file into the assembly and it will be readonly and can not be writable. While you set the file as Copy alwa ...
  • 首先在Form中添加一個控制項,然後將控制項的背景BackColor設置成透明 、 然後分別設置控制項的Image: Image image = Image.FromFile(UserLoginInfor.LoginInfor.userIco); 獲取image的width、height:width= i ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...