前言
在項目中利用Dapper將JSON數(shù)據(jù)存儲到MySQL數(shù)據(jù)庫,結(jié)果發(fā)現(xiàn)JSON數(shù)據(jù)中的中文亂碼,特此記錄,希望對存儲JSON的童鞋能有所幫助,文中若有錯誤之處,還望批評指正。
Dapper獲取JSON數(shù)據(jù)亂碼(MySQL)
為了引出最終問題出在什么地方,我們重頭開始進行講解,首先我們給出如下測試實體以及在數(shù)據(jù)庫中的表,如下:
public class Test { public int Id { get; set; } public string Data { get; set;
} }
為了數(shù)據(jù)操作方便,我們通過包【Dapper.SimpleCRUD】來操作,插入數(shù)據(jù)和查詢數(shù)據(jù)如下:
static void Main(string[] args) {
SimpleCRUD.SetDialect(SimpleCRUD.Dialect.MySQL);
DefaultTypeMap.MatchNamesWithUnderscores= true; using (var conn = new
MySqlConnection(@".....;charset=utf8mb4;SslMode=none;")) { var id = conn.Insert(
new Test() { Data = JsonConvert.SerializeObject(new { name = "汪鵬" }) }); var
result = conn.Get<Test>(id); } Console.ReadKey(); }
如上一切正常,接下來我們將存儲JSON數(shù)據(jù)的列類型修改為json,然后再來進行如上操作,結(jié)果會發(fā)現(xiàn)獲取數(shù)據(jù)中文將亂碼:
?
剛開始我猜想難道用的包【Dapper.SimpleCRUD】在映射時出了問題,于是我用Dapper進行如下查詢,依然會出現(xiàn)如上中文亂碼:
var result = conn.QueryFirstOrDefault<Test>("select * from Test where Id = @id"
,new { id });
?所以由上基本可以得出結(jié)論:針對MySQL中的類型json,若json數(shù)據(jù)中存在中文,則利用Dapper查詢時將出現(xiàn)亂碼。那么如何解決這個問題呢?
于是乎,在添加數(shù)據(jù)時,我將中文進行URL編碼,再查詢時進行解碼即可,如下:
var id = conn.Insert(new Test() { Data = JsonConvert.SerializeObject(new { name
= HttpUtility.UrlEncode("汪鵬") }) });
當(dāng)然,上述解決方案以實際項目業(yè)務(wù)而去解決,我們項目需要獲取到JSON數(shù)據(jù)中的中文然后通過URL傳輸,免去請求接口再去獲取所存儲的名稱。到此我們了解了Dapper針對json數(shù)據(jù)類型將導(dǎo)致查詢中文亂碼的問題,但是其根本原因是什么呢?上述連接MySQL的驅(qū)動,是使用Oracle官方所提供的驅(qū)動且為最新版本,如下:
我們知道通過EF
Core操作MySQL數(shù)據(jù)庫,若是利用官方包有一堆問題存在,所以大多都是采用的包【Pomelo.EntityFrameworkCore.MySql】,因為里面包含【MySqlConnector】,換句話說也就存在MySQL的驅(qū)動連接,所以這里我們嘗試?yán)迷摪鎿Q上述【MySql.Data】包,結(jié)果發(fā)現(xiàn)中文不再亂碼。
總結(jié)
綜上所述,我們需要注意的是
若利用官方驅(qū)動包【MySql.Data】,當(dāng)配置MySQL中的列類型為json時,若json數(shù)據(jù)中存在中文,則利用Dapper查詢時將會出現(xiàn)中文亂碼,需要進行轉(zhuǎn)換
,上述在數(shù)據(jù)庫連接字符串中我們設(shè)置為utf8mb4,所以理論上應(yīng)該和編碼沒有任何關(guān)系,
【最新更新:與Dapper無關(guān),utf8將導(dǎo)致亂碼,需要將表編碼配置為utf8mb4才行】。
熱門工具 換一換