欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

c# Newtonsoft 六個值得使用的特性(下)

 更新時間:2020年06月23日 10:35:24   作者:一線碼農(nóng)  
這篇文章主要介紹了c# Newtonsoft 六個值得使用的特性,文中示例代碼非常詳細(xì),幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下

一:講故事

上一篇介紹的 6 個特性從園子里的反饋來看效果不錯,那這一篇就再帶來 6 個特性同大家一起欣賞。

二:特性分析

1. 像弱類型語言一樣解析 json

大家都知道弱類型的語言有很多,如: nodejs,python,php,它們有一個🐮👃的地方就是處理json,不需要像 強(qiáng)類型語言 那樣還要給它配一個類,什么意思呢? 就拿下面的 json 說事。

{
 "DisplayName": "新一代算法模型",
 "CustomerType": 1,
 "Report": {
 "TotalCustomerCount": 1000,
 "TotalTradeCount": 50
 },
 "CustomerIDHash": [1,2,3,4,5]
}

這個 json 如果想灌到 C# 中處理,你就得給它定義一個適配的類,就如 初篇 的客戶算法模型類,所以這里就有了一個需求,能不能不定義類也可以自由解析上面這串 json 呢??? 哈哈,當(dāng)然是可以的, 反序列化成 Dictionary 即可,就拿提取 Report.TotalCustomerCount CustomerIDHash 這兩個字段演示一下。

  static void Main(string[] args)
  {
   var json = @"{
       'DisplayName': '新一代算法模型',
       'CustomerType': 1,
       'Report': {
        'TotalCustomerCount': 1000,
        'TotalTradeCount': 50
       },
       'CustomerIDHash': [1,2,3,4,5]
       }";

   var dict = JsonConvert.DeserializeObject<Dictionary<object, object>>(json);

   var report = dict["Report"] as JObject;
   var totalCustomerCount = report["TotalCustomerCount"];

   Console.WriteLine($"totalCustomerCount={totalCustomerCount}");

   var arr = dict["CustomerIDHash"] as JArray;
   var list = arr.Select(m => m.Value<int>()).ToList();

   Console.WriteLine($"list={string.Join(",", list)}");
  }

2. 如何讓json中的枚舉保持更易讀的字符串型

這句話是什么意思呢? 默認(rèn)情況下, SerializeObject 會將 Model 中的 Enum 變成數(shù)值型,大家都知道數(shù)值型語義性是非常差的,如下代碼所示:

 static void Main(string[] args)
 {
  var model = new ThreadModel() { ThreadStateEnum = System.Threading.ThreadState.Running };

  var json = JsonConvert.SerializeObject(model);

  Console.WriteLine(json);
 }

 class ThreadModel
 {
  public System.Threading.ThreadState ThreadStateEnum { get; set; }
 }

對吧,確實語義特別差,那能不能直接生成 Running 這種字符串形式呢? 當(dāng)然可以了。。。改造如下:

var json = JsonConvert.SerializeObject(model, new StringEnumConverter());

這里可能就有人鉆牛角尖了,能不能部分指定讓枚舉生成 string,其他的生成 int ,沒關(guān)系,這也難不倒我,哪里使用就用 JsonConverter 標(biāo)記哪里。。。

  static void Main(string[] args)
  {
   var model = new ThreadModel()
   {
    ThreadStateEnum = System.Threading.ThreadState.Running,
    TaskStatusEnum = TaskStatus.RanToCompletion
   };

   var json = JsonConvert.SerializeObject(model);

   Console.WriteLine(json);
  }

  class ThreadModel
  {
   public System.Threading.ThreadState ThreadStateEnum { get; set; }

   [JsonConverter(typeof(StringEnumConverter))]
   public TaskStatus TaskStatusEnum { get; set; }
  } 

3. 格式化 json 中的時間類型

在 model 轉(zhuǎn)化成 json 的過程中,總少不了 時間類型,為了讓時間類型 可讀性更高,通常會 格式化為 YYYY年/MM月/dd日 ,那如何實現(xiàn)呢? 很簡單撒,在 JsonConvert 中也是一個 枚舉 幫你搞定。。。

  static void Main(string[] args)
  {
   var json = JsonConvert.SerializeObject(new Order()
   {
    OrderTitle = "女裝大佬",
    Created = DateTime.Now
   }, new JsonSerializerSettings
   {
    DateFormatString = "yyyy年/MM月/dd日",
   });

   Console.WriteLine(json);
  }
  public class Order
  {
   public string OrderTitle { get; set; }
   public DateTime Created { get; set; }
  } 

對了,我記得很早的時候,C# 自帶了一個 JavaScriptSerializer, 也是用來進(jìn)行 model 轉(zhuǎn) json的,但是它會將 datetime 轉(zhuǎn)成 時間戳,而不是時間字符串形式,如果你因為特殊原因想通過 JsonConvert 將時間生成時間戳的話,也是可以的, 用 DateFormatHandling.MicrosoftDateFormat 枚舉指定一下即可,如下:

4. 對一些常用設(shè)置進(jìn)行全局化

在之前所有演示的特性技巧中都是在 JsonConvert 上指定的,也就是說 100 個 JsonConvert 我就要指定 100 次,那有沒有類似一次指定,整個進(jìn)程通用呢? 這么強(qiáng)大的 Newtonsoft 早就支持啦, 就拿上面的 Order 舉例:

  JsonConvert.DefaultSettings = () =>
  {
   var settings = new JsonSerializerSettings
   {
    Formatting = Formatting.Indented
   };
   return settings;
  };

  var order = new Order() { OrderTitle = "女裝大佬", Created = DateTime.Now };

  var json1 = JsonConvert.SerializeObject(order);
  var json2 = JsonConvert.SerializeObject(order);

  Console.WriteLine(json1);
  Console.WriteLine(json2);

5. 如何保證 json 到 model 的嚴(yán)謹(jǐn)性 及提取 json 未知字段

有時候我們有這樣的需求,一旦 json 中出現(xiàn) model 未知的字段,有兩種選擇: 要么報錯,要么提取出未知字段,在 Newtonsoft 中默認(rèn)的情況是忽略,場景大家可以自己找哈。

  • 未知字段報錯
      static void Main(string[] args)
      {
       var json = "{'OrderTitle':'女裝大佬', 'Created':'2020/6/23','Memo':'訂單備注'}";
    
       var order = JsonConvert.DeserializeObject<Order>(json, new JsonSerializerSettings
       {
        MissingMemberHandling = MissingMemberHandling.Error
       });
    
       Console.WriteLine(order);
      }
    
      public class Order
      {
       public string OrderTitle { get; set; }
       public DateTime Created { get; set; }
       public override string ToString()
       {
        return $"OrderTitle={OrderTitle}, Created={Created}";
       }
      }

  • 提取未知字段

我依稀的記得 WCF 在這種場景下也是使用一個 ExtenstionDataObject 來存儲客戶端傳過來的未知字段,有可能是客戶端的 model 已更新,server端還是舊版本,通常在 json 序列化中也會遇到這種情況,在 JsonConvert 中使用 _additionalData 就可以幫你搞定,在 OnDeserialized 這種AOP方法中進(jìn)行攔截,如下代碼:

 static void Main(string[] args)
 {
  var json = "{'OrderTitle':'女裝大佬', 'Created':'2020/6/23','Memo':'訂單備注'}";

  var order = JsonConvert.DeserializeObject<Order>(json);

  Console.WriteLine(order);
 }

 public class Order
 {
  public string OrderTitle { get; set; }

  public DateTime Created { get; set; }

  [JsonExtensionData]
  private IDictionary<string, JToken> _additionalData;

  public Order()
  {
   _additionalData = new Dictionary<string, JToken>();
  }

  [OnDeserialized]
  private void OnDeserialized(StreamingContext context)
  {
   var dict = _additionalData;
  }

  public override string ToString()
  {
   return $"OrderTitle={OrderTitle}, Created={Created}";
  }
 } 

6. 開啟 JsonConvert 詳細(xì)日志功能

有時候在查閱源碼的時候開啟日志功能更加有利于理解源碼的內(nèi)部運(yùn)作,所以這也是一個非常實用的功能,看看如何配置吧。

  static void Main(string[] args)
  {
   var json = "{'OrderTitle':'女裝大佬', 'Created':'2020/6/23','Memo':'訂單備注'}";

   MemoryTraceWriter traceWriter = new MemoryTraceWriter();

   var account = JsonConvert.DeserializeObject<Order>(json, new JsonSerializerSettings
   {
    TraceWriter = traceWriter
   });

   Console.WriteLine(traceWriter.ToString());
  }

  public class Order
  {
   public string OrderTitle { get; set; }

   public DateTime Created { get; set; }

   public override string ToString()
   {
    return $"OrderTitle={OrderTitle}, Created={Created}";
   }
  }

三:總結(jié)

嘿嘿,這篇 6 個特性就算說完了, 結(jié)合上一篇一共 12 個特性,是不是非常簡單且實用,后面準(zhǔn)備給大家?guī)硪恍┰创a解讀吧! 希望本篇對您有幫助,謝謝!

以上就是c# Newtonsoft 六個值得使用的特性(下)的詳細(xì)內(nèi)容,更多關(guān)于c# Newtonsoft 特性的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 使用C#創(chuàng)建Windows服務(wù)的實例代碼

    使用C#創(chuàng)建Windows服務(wù)的實例代碼

    這篇文章主要介紹了使用C#創(chuàng)建Windows服務(wù)的實例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-07-07
  • C#實現(xiàn)的簡單驗證碼識別實例

    C#實現(xiàn)的簡單驗證碼識別實例

    這篇文章主要介紹了C#實現(xiàn)的簡單驗證碼識別實例,只適應(yīng)一些簡單的驗證碼,需要的朋友可以參考下
    2014-06-06
  • C#獲取進(jìn)程和對進(jìn)程的操作

    C#獲取進(jìn)程和對進(jìn)程的操作

    下面是一個例子:獲取進(jìn)程列表、創(chuàng)建“違禁”進(jìn)程名單、查找并終止進(jìn)程,注意先要在項目里添加System.Management的引用
    2013-04-04
  • 淺談C#索引器

    淺談C#索引器

    這篇文章主要簡單介紹C#索引器,索引器使你可從語法上方便地創(chuàng)建類、結(jié)構(gòu)或接口,以便客戶端應(yīng)用程序可以像訪問數(shù)組一樣訪問它們。編譯器將生成一個 Item 屬性和適當(dāng)?shù)脑L問器方法,在主要目標(biāo)是封裝內(nèi)部集合或數(shù)組的類型中,常常要實現(xiàn)索引器,下面我們一起來看看具體內(nèi)容吧
    2021-11-11
  • C#實現(xiàn)讀取二維數(shù)組集合并輸出到Word預(yù)設(shè)表格

    C#實現(xiàn)讀取二維數(shù)組集合并輸出到Word預(yù)設(shè)表格

    這篇文章主要為大家詳細(xì)介紹了如何使用C#實現(xiàn)讀取二維數(shù)組集合并輸出到Word預(yù)設(shè)表格,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2024-03-03
  • 如何從dump文件中提取出C#源代碼

    如何從dump文件中提取出C#源代碼

    這篇文章主要介紹了如何從dump文件中提取出C#源代碼,幫助大家更好的理解和學(xué)習(xí)使用c#,感興趣的朋友可以了解下
    2021-03-03
  • c# record的使用場景

    c# record的使用場景

    這篇文章主要介紹了c# record的使用場景,幫助大家更好的理解和學(xué)習(xí)使用c#的新特性,感興趣的朋友可以了解下
    2021-02-02
  • C#判斷DLL文件是32位還是64位的示例代碼

    C#判斷DLL文件是32位還是64位的示例代碼

    有些時候我們需要判斷一下dll文件是32位還是64位,糾結(jié)該如何操作呢,下面小編通過實例代碼給大家介紹下C#判斷DLL文件是32位還是64位,感興趣的朋友跟隨小編一起看看吧
    2021-12-12
  • C#中的multipart/form-data提交文件和參數(shù)

    C#中的multipart/form-data提交文件和參數(shù)

    這篇文章主要介紹了C#中的multipart/form-data提交文件和參數(shù),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-06-06
  • C#實現(xiàn)XML與實體類之間相互轉(zhuǎn)換的方法(序列化與反序列化)

    C#實現(xiàn)XML與實體類之間相互轉(zhuǎn)換的方法(序列化與反序列化)

    這篇文章主要介紹了C#實現(xiàn)XML與實體類之間相互轉(zhuǎn)換的方法,涉及C#序列化與反序列化操作的相關(guān)技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2016-06-06

最新評論