C# 開(kāi)發(fā)日志本地化工具
程序員討厭寫(xiě)文檔, 討厭寫(xiě)注釋, 而我還討厭寫(xiě)日志, 輸出一個(gè) "Id=5, 姓名=王大錘, 性別=男, 生日=2020年1月1日" 總歸會(huì)用到字符串的填充
var log = $"Id={person.Id}, 姓名={person.Name}, 性別={(person.Sex == SexType.Man ? "男性" : "女性")}, 生日={person.Birthday}";
Json序列化工具多好啊, 可是輸出的是
{"id": 5,"name":"葫蘆娃", "sex":"Man", "birthday":"2020-1-1 00:00:00"}
業(yè)務(wù)部門(mén)的人就是看不懂, 畢竟不是人人都有良好的英語(yǔ)基礎(chǔ), 同時(shí)我也經(jīng)常猜不到有人用 DRLS 表示 "當(dāng)日流水".
其實(shí)如果只要稍微把 json 里面的key 用中文替代, 業(yè)務(wù)部門(mén)還是能大概讀得懂大部分意思的.
所以我開(kāi)發(fā)了一個(gè)工具 LocalizationTools, 協(xié)助生成中文日志.
新建一個(gè) Console, 引入 nuget 包: LocalizationTools, 然后定義示例類
/// <summary>
/// 人類
/// </summary>
public class Person
{
/// <summary>
/// Id
/// </summary>
public int Id { get; set; }
/// <summary>
/// 名字
/// </summary>
public string Name { get; set; }
/// <summary>
/// 出生日期, 出生日期最好不要超過(guò)當(dāng)前時(shí)間
/// </summary>
[DisplayName("出生日期")]
public DateTime Birthday { get; set; }
/// <summary>
/// 性別
/// </summary>
public SexType Sex { get; set; }
/// <summary>
/// 是否活著
/// </summary>
public bool IsAlive { get; set; }
}
/// <summary>
/// 性別
/// </summary>
public enum SexType
{
/// <summary>
/// 男性
/// </summary>
Man = 0,
/// <summary>
/// 女性
/// </summary>
Woman = 2,
/// <summary>
/// 人妖
/// </summary>
Ladyman = 3,
}
記得在生成界面勾上 XML文檔文件

使用代碼
static void Main(string[] args)
{
var p1 = new Person
{
Id = 1,
Name = "王大錘",
Birthday = DateTime.Parse("2020-01-01"),
Sex = SexType.Man,
};
LocalizationTools.KeyValueSeparator = "=";
var str = LocalizationTools.ToString(p1);
Console.WriteLine(str);
}
相信這樣的輸出, 大部分人也應(yīng)該能夠看懂了
{"Id"=1,"名字"="王大錘","出生日期"="2020/1/1 0:00:00","性別"="男性","是否活著"=false}
LocalizationTools.ToString() 方法會(huì)將 屬性名稱 替換成注釋里的 Summary 信息, 枚舉值也同樣會(huì)進(jìn)行這樣的替換
如果字段很少, 剛才的輸出還沒(méi)什么問(wèn)題, 如果字段非常多, 讀著就眼花繚亂了, 所以我建議還是這行刪除
LocalizationTools.KeyValueSeparator = "=";
這樣輸出的內(nèi)容是
{"Id":1,"名字":"王大錘","出生日期":"2020/1/1 0:00:00","性別":"男性","是否活著":false}
使用Json工具格式化一下
{
"Id": 1,
"名字": "王大錘",
"出生日期": "2020/1/1 0:00:00",
"性別": "男性",
"是否活著": false
}
這樣即使包含了子對(duì)象的對(duì)象, 也非常清晰明了了.
這里面的不足是: "是否活著" 這個(gè)屬性輸出的是 true/false, 布爾值在不同的場(chǎng)景可以表示: 是/否、對(duì)/錯(cuò)、啟用/關(guān)閉....... 業(yè)務(wù)人員可不想自己猜, 解決辦法有兩個(gè)
1. 在ToString()前, 我知道IsAlive是false, 應(yīng)該用 "否" 來(lái)替換
var str = LocalizationTools.ToString(p1, new { IsAlive = "否" });
// 輸出 {"Id":1,"名字":"王大錘","出生日期":"2020/1/1 0:00:00","性別":"男性","是否活著":"否"}
2. 給 IsAlive 屬性加上 ToStringReplacePairAttribute, 來(lái)替換某些特定的值
[ToStringReplacePair(true, "是", false, "否")]
public bool IsAlive { get; set; }
LocalizationTools 替換 屬性名稱 的順序是 1. DisplayNameAttribute 2. 注釋里的summary, 因?yàn)橛行┤讼矚g在 summary 中加入其他說(shuō)明信息, 輸出到日志里不好看;
由于 DisplayNameAttribute 不能作用于 enum枚舉值, 所以我專門(mén)定義了 EnumAliasAttribute, 它的優(yōu)先級(jí)也比 注釋里的summary 高
這里特別強(qiáng)調(diào)一下, LocalizationTools.ToString() 不是一個(gè)Json 序列化工具, 為了使用隨處可見(jiàn)的 Json格式化工具, 而將輸出調(diào)整得像Json, 所以這個(gè)工具從來(lái)就沒(méi)有考慮到反序列化功能, 也沒(méi)有去解決循環(huán)引用的問(wèn)題, 也沒(méi)有考慮到要符合Json 的標(biāo)準(zhǔn), 僅僅是一個(gè)方便輸出中文日志的工具, 也沒(méi)有追求高性能.
LocalizationTools.ToString() 特別適用于面向數(shù)據(jù)表的編程, 因?yàn)楸碜侄我话愣际呛?jiǎn)單的類型, 輸出的日志更為直觀.
按理說(shuō)應(yīng)該為這個(gè)工具提供擴(kuò)展方法, 但是我有強(qiáng)迫癥, 許多類庫(kù)給 object 加上了各種擴(kuò)展方法, 讓我很不爽, 所以我沒(méi)有在類庫(kù)中主動(dòng)加入擴(kuò)展方法, 大家可以在自己的項(xiàng)目里加入以下代碼, 以提供擴(kuò)展方法
namespace Localization
{
using System.Collections.Generic;
public static class LocalizationToolsExtend
{
public static string ToLocalizationString(this object obj, params string[] ignorePropertyNames)
{
return LocalizationTools.ToString(obj, ignorePropertyNames);
}
public static string ToLocalizationString<T>(this object obj, T customPropertyValues, params string[] ignorePropertyNames)
where T : class
{
return LocalizationTools.ToString(obj, customPropertyValues, ignorePropertyNames);
}
public static string ToLocalizationString(this object obj, Dictionary<string, object> customPropertyValues, params string[] ignorePropertyNames)
{
return LocalizationTools.ToString(obj, customPropertyValues, ignorePropertyNames);
}
public static string ToLocalizationStringInclude(this object obj, IEnumerable<string> includePropertyNames)
{
return LocalizationTools.ToStringInclude(obj, includePropertyNames);
}
public static string ToLocalizationStringInclude<T>(this object obj, IEnumerable<string> includePropertyNames, T customPropertyValues)
where T : class
{
return LocalizationTools.ToStringInclude(obj, includePropertyNames, customPropertyValues);
}
public static string ToLocalizationStringInclude(this object obj, IEnumerable<string> includePropertyNames, Dictionary<string, object> customPropertyValues)
{
return LocalizationTools.ToStringInclude(obj, includePropertyNames, customPropertyValues);
}
}
}
新增實(shí)體的日志解決了, 接下來(lái)又有另一個(gè)問(wèn)題, 如何保存實(shí)體變化的日志?
最簡(jiǎn)單的辦法就是把 實(shí)體類 修改前的json 和 修改后的json 都保存起來(lái), 讓業(yè)務(wù)人員自己去痛苦尋找的變化, 呵呵呵, 只要是個(gè)人, 都會(huì)抱怨.
讓程序員一個(gè)字段一個(gè)字段的比較, 然后生成日志, 開(kāi)玩笑! 我干不來(lái)這樣枯燥的活
nuget 上有 JsonDiffPatch 這樣的工具生成 JSON patch, 但輸出的結(jié)果就不是為人類準(zhǔn)備的, 所以我又繼續(xù)寫(xiě)了 Compare 方法
static void Main(string[] args)
{
var p1 = new Person
{
Id = 1,
Name = "王大錘",
Birthday = DateTime.Parse("2020-01-01"),
Sex = SexType.Man,
};
var p2 = new Person
{
Id = 1,
Name = "王小錘",
Birthday = DateTime.Parse("2021-01-01"),
Sex = SexType.Man,
};
var compareResult = LocalizationTools.Compare(p1, p2);
Console.WriteLine(compareResult.GetDifferenceMsg());
}
// 輸出: {"名字":{"從":"王大錘","變成":"王小錘"},"出生日期":{"從":"2020/1/1 0:00:00","變成":"2021/1/1 0:00:00"}}
Json格式化一下
{
"名字": {
"從": "王大錘",
"變成": "王小錘"
},
"出生日期": {
"從": "2020/1/1 0:00:00",
"變成": "2021/1/1 0:00:00"
}
}
當(dāng)然你可以對(duì) CompareResult 進(jìn)行進(jìn)一步處理, 使用 UpdateDifferentProperty 修改里面的比較結(jié)果, 最后再得出比較結(jié)果.
以上就是C# 開(kāi)發(fā)日志本地化工具的詳細(xì)內(nèi)容,更多關(guān)于C# 日志本地化工具的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- 基于c# Task自己動(dòng)手寫(xiě)個(gè)異步IO函數(shù)
- C#異步方法返回void與Task的區(qū)別詳解
- 深入分析C#中的異步和多線程
- c# winform異步不卡界面的實(shí)現(xiàn)方法
- C#用委托BeginInvoke做異步線程
- C#中一個(gè)高性能異步socket封裝庫(kù)的實(shí)現(xiàn)思路分享
- C#實(shí)現(xiàn)異步編程的方法
- c#中Winform實(shí)現(xiàn)多線程異步更新UI(進(jìn)度及狀態(tài)信息)
- c# 用Dictionary實(shí)現(xiàn)日志數(shù)據(jù)批量插入
- c# 用ELMAH日志組件處理異常
- C#使用SqlServer作為日志數(shù)據(jù)庫(kù)的設(shè)計(jì)與實(shí)現(xiàn)
- C#打印日志的方法總結(jié)
- c#快速寫(xiě)本地日志方法
- C#中四步輕松使用log4net記錄本地日志的方法
- c# 編寫(xiě)一個(gè)輕量級(jí)的異步寫(xiě)日志的實(shí)用工具類(LogAsyncWriter)
相關(guān)文章
C#讀取xml節(jié)點(diǎn)數(shù)據(jù)方法小結(jié)
這篇文章主要介紹了C#讀取xml節(jié)點(diǎn)數(shù)據(jù)的方法,實(shí)例總結(jié)了C#針對(duì)XML文件節(jié)點(diǎn)操作的相關(guān)技巧,需要的朋友可以參考下2015-06-06
C#和SQL實(shí)現(xiàn)的字符串相似度計(jì)算代碼分享
這篇文章主要介紹了C#和SQL實(shí)現(xiàn)的字符串相似度計(jì)算代碼分享,本文分別給出了C#語(yǔ)言和SQL語(yǔ)言的實(shí)現(xiàn)代碼,需要的朋友可以參考下2014-10-10
winform樹(shù)形菜單無(wú)限級(jí)分類實(shí)例
本文介紹了“winform樹(shù)形菜單無(wú)限級(jí)分類實(shí)例”,需要的朋友可以參考一下2013-03-03

