c# Newtonsoft 六個值得使用的特性(下)
一:講故事
上一篇介紹的 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ù)的實例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-07-07C#實現(xiàn)讀取二維數(shù)組集合并輸出到Word預(yù)設(shè)表格
這篇文章主要為大家詳細(xì)介紹了如何使用C#實現(xiàn)讀取二維數(shù)組集合并輸出到Word預(yù)設(shè)表格,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-03-03C#中的multipart/form-data提交文件和參數(shù)
這篇文章主要介紹了C#中的multipart/form-data提交文件和參數(shù),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-06-06C#實現(xiàn)XML與實體類之間相互轉(zhuǎn)換的方法(序列化與反序列化)
這篇文章主要介紹了C#實現(xiàn)XML與實體類之間相互轉(zhuǎn)換的方法,涉及C#序列化與反序列化操作的相關(guān)技巧,具有一定參考借鑒價值,需要的朋友可以參考下2016-06-06