一:講故事
上一篇介紹的 6 個特性從園子里的反饋來看效果不錯,那這一篇就再帶來 6 個特性同大家一起欣賞,
二:特性分析
1. 像弱型別語言一樣決議 json
大家都知道弱型別的語言有很多,如: nodejs,python,php,它們有一個????的地方就是處理json,不需要像 強型別語言 那樣還要給它配一個類,什么意思呢? 就拿下面的 json 說事,
{
"DisplayName": "新一代演算法模型",
"CustomerType": 1,
"Report": {
"TotalCustomerCount": 1000,
"TotalTradeCount": 50
},
"CustomerIDHash": [1,2,3,4,5]
}
這個 json 如果想灌到 C# 中處理,你就得給它定義一個適配的類,就如 初篇 的客戶演算法模型類,所以這里就有了一個需求,能不能不定義類也可以自由決議上面這串 json 呢??? 哈哈,當然是可以的, 反序列化成 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中的列舉保持更易讀的字串型
這句話是什么意思呢? 默認情況下, SerializeObject 會將 Model 中的 Enum 變成數值型,大家都知道數值型語意性是非常差的,如下代碼所示:
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 這種字串形式呢? 當然可以了,,,改造如下:
var json = JsonConvert.SerializeObject(model, new StringEnumConverter());

這里可能就有人鉆牛角尖了,能不能部分指定讓列舉生成 string,其他的生成 int ,沒關系,這也難不倒我,哪里使用就用 JsonConverter 標記哪里,,,
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 轉化成 json 的程序中,總少不了 時間型別,為了讓時間型別 可讀性更高,通常會 格式化為 YYYY年/MM月/dd日 ,那如何實作呢? 很簡單撒,在 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, 也是用來進行 model 轉 json的,但是它會將 datetime 轉成 時間戳,而不是時間字串形式,如果你因為特殊原因想通過 JsonConvert 將時間生成時間戳的話,也是可以的, 用 DateFormatHandling.MicrosoftDateFormat 列舉指定一下即可,如下:

4. 對一些常用設定進行全域化
在之前所有演示的特性技巧中都是在 JsonConvert 上指定的,也就是說 100 個 JsonConvert 我就要指定 100 次,那有沒有類似一次指定,整個行程通用呢? 這么強大的 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);

可以看到,Formatting.Indented 對兩串 json 都生效了,
5. 如何保證 json 到 model 的嚴謹性 及提取 json 未知欄位
有時候我們有這樣的需求,一旦 json 中出現 model 未知的欄位,有兩種選擇: 要么報錯,要么提取出未知欄位,在 Newtonsoft 中默認的情況是忽略,場景大家可以自己找哈,
- 未知欄位報錯
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 序列化中也會遇到這種情況,這里只要使用 JsonExtensionData 特性就可以幫你搞定,在 OnDeserialized 這種AOP方法中進行攔截,如下代碼:
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 = https://www.cnblogs.com/huangxincheng/p/new Dictionary();
}
[OnDeserialized]
private void OnDeserialized(StreamingContext context)
{
var dict = _additionalData;
}
public override string ToString()
{
return $"OrderTitle={OrderTitle}, Created={Created}";
}
}

6. 開啟 JsonConvert 詳細日志功能
有時候在查閱原始碼的時候開啟日志功能更加有利于理解原始碼的內部運作,所以這也是一個非常實用的功能,看看如何配置吧,
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}";
}
}

三:總結
嘿嘿,這篇 6 個特性就算說完了, 結合上一篇一共 12 個特性,是不是非常簡單且實用,后面準備給大家帶來一些原始碼解讀吧! 希望本篇對您有幫助,謝謝!
如您有更多問題與我互動,掃描下方進來吧~
轉載請註明出處,本文鏈接:https://www.uj5u.com/net/13495.html
標籤:C#
