狠狠色丁香婷婷综合尤物/久久精品综合一区二区三区/中国有色金属学报/国产日韩欧美在线观看 - 国产一区二区三区四区五区tv

LOGO OA教程 ERP教程 模切知識交流 PMS教程 CRM教程 開發(fā)文檔 其他文檔  
 
網(wǎng)站管理員

巧用 JSON序列化/反序列化,輕松實(shí)現(xiàn)C#對象映射

admin
2024年12月19日 17:51 本文熱度 379

一、引言

簡述

?C#編程的世界里,對象映射是一項(xiàng)十分重要且常用的操作,而利用JSON序列化/反序列化來實(shí)現(xiàn)對象映射更是有著廣泛的應(yīng)用場景。比如說,當(dāng)我們需要與外部系統(tǒng)進(jìn)行數(shù)據(jù)交互時,常常要把C#中的對象轉(zhuǎn)換為JSON格式的數(shù)據(jù)發(fā)送出去,這就涉及到對象的序列化;而接收到外部傳來的JSON數(shù)據(jù)后,又需要將其還原成C#對象以便后續(xù)處理,這就是反序列化的過程了。再比如,在存儲數(shù)據(jù)或者對數(shù)據(jù)進(jìn)行傳輸封裝等操作時,這種基于JSON的對象映射方式也能幫我們更便捷地處理數(shù)據(jù)。

今天,我就來給大家詳細(xì)分享一下利用?JSON序列化/反序列化輕松實(shí)現(xiàn)對象映射的具體步驟,相信掌握了這些內(nèi)容,會讓大家在面對相關(guān)開發(fā)需求時更加得心應(yīng)手,下面咱們就正式開始吧。

二、JSON序列化/反序列化基礎(chǔ)知識

JSON數(shù)據(jù)格式簡介

JSONJavaScript Object Notation)是一種輕量級的數(shù)據(jù)交換格式,它基于JavaScript編程語言的一個子集,但因其文本格式清晰簡潔,被廣泛用于數(shù)據(jù)交換和存儲。

JSON所支持的數(shù)據(jù)結(jié)構(gòu)及常見表現(xiàn)形式如下:

?對象(Object:對象是由鍵值對組成的無序集合,鍵是字符串,值可以是任何類型,包括對象和數(shù)組。對象由一對花括號{ } 包圍,鍵和值之間用冒號: 分隔,鍵值對之間用逗號, 分隔。例如:{"name": "John Doe", "age": 30, "isMarried": false, "children": ["Alice", "Bob"]}。需要注意的是,JSON中對象的屬性名必須用雙引號,屬性值如果是字符串也必須用雙引號,只要涉及到字符串,就必須使用雙引號,并且不支持undefined

?數(shù)組(Array:數(shù)組是值的有序集合,每個值可以是任何類型,包括對象和數(shù)組。數(shù)組由一對方括號[ ] 包裹,值之間用逗號, 分隔。比如:["apple", "banana", "cherry"]

?字符串(StringJSON字符串是由雙引號" " 包圍的?Unicode字符序列,使用反斜杠\ 轉(zhuǎn)義,像{"name": "Apifox", "city": "Guangzhou"} 中的"name" "city" 就是字符串的示例。

?數(shù)字(Number:可以是整數(shù)或浮點(diǎn)數(shù),例如{"age": 30, "weight": 65.5} 中的30 65.5 分別是整數(shù)和浮點(diǎn)數(shù),數(shù)字不需要雙引號包圍。

?布爾值(Boolean:只有兩個取值,即true false,如{"isStudent": true, "isEmployed": false} 中的isStudent isEmployed 取值示例,同樣不需要雙引號包圍。

?null:表示一個空值,像{"middleName": null} 里的middleName 的值就是null,也無需雙引號包圍。

這些結(jié)構(gòu)還可以相互嵌套,形成復(fù)雜的?JSON數(shù)據(jù)格式,以此來滿足不同場景下的數(shù)據(jù)表示需求。

C#中相關(guān)命名空間及類庫介紹

?C#里,常用于JSON序列化/反序列化操作的有以下幾個重要的命名空間及對應(yīng)的類庫:

1.System.Web.Script.Serialization

這個命名空間下的JavaScriptSerializer 類可用于?JSON的序列化和反序列化。例如,以下是簡單的代碼示例展示其使用方式:

using System;

using System.Web.Script.Serialization;

public class Program

{

? ??public static void Main()

? ??{

? ? ? ??//創(chuàng)建一個示例對象

? ? ? ??var person = new Person { Name = "張三", Age = 30 };

? ? ? ??//序列化對象到JSON字符串

? ? ? ??JavaScriptSerializer serializer = new JavaScriptSerializer();

? ? ? ??string json = serializer.Serialize(person);

? ? ? ??Console.WriteLine(json);

? ? ? ??//反序列化JSON字符串到對象

? ? ? ??string jsonString = "{\"Name\":\"李四\",\"Age\":25}";

? ? ? ??Person deserializedPerson = serializer.Deserialize<Person>(jsonString);

? ? ? ??Console.WriteLine($"Name: {deserializedPerson.Name}, Age: {deserializedPerson.Age}");

? ??}

? ??public class Person

? ??{

? ? ? ??public string Name { get; set; }

? ? ? ??public int Age { get; set; }

? ??}

}

不過,該類在處理DateTime 類型數(shù)據(jù)的序列化時可能存在一些小問題,比如序列化后的時間可能會出現(xiàn)偏差等情況。

2.Newtonsoft.JsonJSON.NET

這是一個成熟且使用廣泛的第三方庫,擁有豐富的功能和強(qiáng)大的靈活性,多年來在.NET社區(qū)中被廣泛接受和使用。它支持弱類型轉(zhuǎn)換,并且提供了一些更高級的功能,比如完全自定義的序列化和反序列化邏輯,能處理循環(huán)引用、自定義轉(zhuǎn)換器、忽略屬性等等,可以方便地應(yīng)對一些復(fù)雜的JSON數(shù)據(jù)場景。使用前需要通過NuGet進(jìn)行安裝,示例代碼如下:

using Newtonsoft.Json;

public class Person

{

? ??public string Name { get; set; }

? ??public int Age { get; set; }

}

Person person = new Person { Name = "John Doe", Age = 30 };

//序列化

string json = JsonConvert.SerializeObject(person);

Console.WriteLine(json);

//反序列化

string data = "{\"Name\":\"李四\",\"Age\":25}";

person = JsonConvert.DeserializeObject<Person>(data);

3.System.Text.Json

它是.NET Core的一部分,在創(chuàng)建跨平臺應(yīng)用程序時,不需要額外的依賴項(xiàng),使用起來較為方便。并且在性能方面進(jìn)行了優(yōu)化,通常比Newtonsoft.Json 更快,利用了新的讀寫?API,采用更高效的內(nèi)部實(shí)現(xiàn),提供了更好的性能和內(nèi)存利用率。同時,它還提供了一些簡化的API,在處理簡單的JSON數(shù)據(jù)時更容易操作,不過默認(rèn)是強(qiáng)類型轉(zhuǎn)換,比如實(shí)體類定義的是字符串,JSON字符串返回的是整型,轉(zhuǎn)換時會報錯。示例代碼如下:

using System;

using System.Text.Json;

namespace SerializeJsonExample

{

? ??class Program

? ??{

? ? ? ??static void Main(string[] args)

? ? ? ??{

? ? ? ? ? ??//創(chuàng)建一個示例對象

? ? ? ? ? ??var person = new Person { Name = "張三", Age = 30 };

? ? ? ? ? ??//使用JsonSerializer.Serialize方法將對象序列化為JSON字符串

? ? ? ? ? ??string jsonString = JsonSerializer.Serialize(person);

? ? ? ? ? ??Console.WriteLine(jsonString);

? ? ? ? ? ??//使用JsonSerializer.Deserialize方法將Json字符串反序列化為對象

? ? ? ? ? ??string data = "{\"Name\":\"李四\",\"Age\":25}";

? ? ? ? ? ??Person p = JsonSerializer.Deserialize<Person>(data);

? ? ? ??}

? ? ? ??//定義一個簡單的Person類,用于序列化示例

? ? ? ??public class Person

? ? ? ??{

? ? ? ? ? ??public string Name { get; set; }

? ? ? ? ? ??public int Age { get; set; }

? ? ? ??}

? ??}

}

開發(fā)者可以根據(jù)具體的項(xiàng)目需求、數(shù)據(jù)復(fù)雜程度以及性能等方面的考量,來選擇合適的命名空間及類庫進(jìn)行?JSON序列化/反序列化操作。

三、利用?System.Runtime.Serialization.Json實(shí)現(xiàn)對象映射

類的準(zhǔn)備與標(biāo)記

當(dāng)我們使用System.Runtime.Serialization.Json 這個命名空間來進(jìn)行對象映射相關(guān)操作時,有個重要的前提要求,那就是參與序列化和反序列化的類需要進(jìn)行特定的標(biāo)記。

具體來說,類需要標(biāo)記為[DataContract],它相當(dāng)于一種契約,是使用DataContractJsonSerializer 序列化和反序列化必須要添加的標(biāo)識。而對應(yīng)類中的屬性,則要標(biāo)記為[DataMember],只有標(biāo)記了的屬性才會參與到序列化和反序列化的過程當(dāng)中。

以下面這個簡單的Student 類為例來看看具體如何標(biāo)記:

using System.Runtime.Serialization;

namespace JsonSerializerAndDeSerializer

{

? ??[DataContract]

? ??public class Student

? ??{

? ? ? ??[DataMember]

? ? ? ??public int ID { get; set; }

? ? ? ??[DataMember]

? ? ? ??public string Name { get; set; }

? ? ? ??[DataMember]

? ? ? ??public int Age { get; set; }

? ? ? ??[DataMember]

? ? ? ??public string Sex { get; set; }

? ??}

}

在上述代碼中,Student 類標(biāo)記了[DataContract],其內(nèi)部的IDNameAgeSex 這幾個屬性都標(biāo)記了[DataMember],這樣在后續(xù)利用System.Runtime.Serialization.Json 相關(guān)功能時,這個類的對象就可以順利地進(jìn)行序列化和反序列化操作了,將類和屬性按照這樣的要求標(biāo)記好,是利用該方式實(shí)現(xiàn)對象映射的基礎(chǔ)準(zhǔn)備工作哦。

序列化操作步驟

接下來詳細(xì)講講利用DataContractJsonSerializer 類將?C#對象序列化為JSON字符串的具體流程。

第一步,我們需要先創(chuàng)建好要進(jìn)行序列化的對象。比如還是以剛才定義的Student 類為例,我們可以這樣創(chuàng)建一個Student 類的實(shí)例:

Student stu = new Student()

{

? ??ID = 01,

? ??Name = "釘釘",

? ??Sex = "",

? ??Age = 1

};

第二步,實(shí)例化DataContractJsonSerializer 序列化器,這里要傳入我們想要序列化的對象類型,代碼如下:

DataContractJsonSerializer js = new DataContractJsonSerializer(typeof(Student));

第三步,創(chuàng)建一個MemoryStream 內(nèi)存流對象,用于后續(xù)將序列化之后的?JSON格式數(shù)據(jù)寫入到這個流當(dāng)中,像這樣:

MemoryStream msObj = new MemoryStream();

第四步,利用剛才實(shí)例化好的序列化器的WriteObject 方法,把我們創(chuàng)建的對象寫入到內(nèi)存流中,對應(yīng)代碼為:

js.WriteObject(msObj, stu);

第五步,要從內(nèi)存流中讀取數(shù)據(jù),在讀取之前需要先將內(nèi)存流的位置設(shè)置為?0,也就是起始位置,然后創(chuàng)建StreamReader 讀取流中的數(shù)據(jù),并將讀取到的內(nèi)容轉(zhuǎn)換為字符串,代碼示例如下:

msObj.Position = 0;

StreamReader sr = new StreamReader(msObj, Encoding.UTF8);

string json = sr.ReadToEnd();

sr.Close();

msObj.Close();

最后,我們就可以得到序列化后的?JSON字符串了。完整的代碼整合起來就是這樣:

using System;

using System.IO;

using System.Runtime.Serialization.Json;

using System.Text;

namespace JsonSerializerAndDeSerializer

{

? ??class Program

? ??{

? ? ? ??static void Main()

? ? ? ??{

? ? ? ? ? ??Student stu = new Student()

? ? ? ? ? ??{

? ? ? ? ? ? ? ??ID = 01,

? ? ? ? ? ? ? ??Name = "釘釘",

? ? ? ? ? ? ? ??Sex = "",

? ? ? ? ? ? ? ??Age = 1

? ? ? ? ? ??};

? ? ? ? ? ??DataContractJsonSerializer js = new DataContractJsonSerializer(typeof(Student));

? ? ? ? ? ??MemoryStream msObj = new MemoryStream();

? ? ? ? ? ??js.WriteObject(msObj, stu);

? ? ? ? ? ??msObj.Position = 0;

? ? ? ? ? ??StreamReader sr = new StreamReader(msObj, Encoding.UTF8);

? ? ? ? ? ??string json = sr.ReadToEnd();

? ? ? ? ? ??sr.Close();

? ? ? ? ? ??msObj.Close();

? ? ? ? ? ??Console.WriteLine(json);

? ? ? ??}

? ??}

? ??[DataContract]

? ??public class Student

? ??{

? ? ? ??[DataMember]

? ? ? ??public int ID { get; set; }

? ? ? ??[DataMember]

? ? ? ??public string Name { get; set; }

? ? ? ??[DataMember]

? ? ? ??public int Age { get; set; }

? ? ? ??[DataMember]

? ? ? ??public string Sex { get; set; }

? ??}

}

假如上述代碼中Student 類對象的各個屬性值分別是ID = 01Name = "釘釘"Sex = ""Age = 1,那最終序列化得到的?JSON字符串結(jié)果可能類似這樣:{"ID":1,"Name":"釘釘","Age":1,"Sex":""}(實(shí)際結(jié)果會根據(jù)具體類結(jié)構(gòu)和屬性值情況有所不同哦),這樣就完成了從?C#對象到JSON字符串的序列化操作啦。

反序列化操作步驟

下面講解將?JSON字符串反序列化為C#對象的具體操作步驟。

首先,我們得有一個?JSON字符串,這個字符串可以是從外部獲取到的數(shù)據(jù),比如從網(wǎng)絡(luò)接口返回的JSON格式的數(shù)據(jù)內(nèi)容等,假設(shè)我們現(xiàn)在已經(jīng)有了這樣一個JSON字符串變量toDes,示例如下:

string toDes = "{\"ID\":1,\"Name\":\"釘釘\",\"Age\":1,\"Sex\":\"\"}";

第一步,要實(shí)例化DataContractJsonSerializer 反序列化器,同樣需要傳入要反序列化的目標(biāo)對象類型,代碼如下:

using (var ms = new MemoryStream(Encoding.Unicode.GetBytes(toDes)))

{

? ??DataContractJsonSerializer deseralizer = new DataContractJsonSerializer(typeof(Student));

第二步,利用反序列化器的ReadObject 方法,從內(nèi)存流中讀取數(shù)據(jù)并還原成對應(yīng)的?C#對象,像這樣:

Student model = (Student)deseralizer.ReadObject(ms);

整合起來完整的反序列化代碼示例如下:

using System;

using System.IO;

using System.Runtime.Serialization.Json;

using System.Text;

namespace JsonSerializerAndDeSerializer

{

? ??class Program

? ??{

? ? ? ??static void Main()

? ? ? ??{

? ? ? ? ? ??string toDes = "{\"ID\":1,\"Name\":\"釘釘\",\"Age\":1,\"Sex\":\"\"}";

? ? ? ? ? ??using (var ms = new MemoryStream(Encoding.Unicode.GetBytes(toDes)))

? ? ? ? ? ??{

? ? ? ? ? ? ? ??DataContractJsonSerializer deseralizer = new DataContractJsonSerializer(typeof(Student));

? ? ? ? ? ? ? ??Student model = (Student)deseralizer.ReadObject(ms);

? ? ? ? ? ? ? ??Console.WriteLine($"ID: {model.ID}, Name: {model.Name}, Age: {model.Age}, Sex: {model.Sex}");

? ? ? ? ? ??}

? ? ? ??}

? ??}

? ??[DataContract]

? ??public class Student

? ??{

? ? ? ??[DataMember]

? ? ? ??public int ID { get; set; }

? ? ? ??[DataMember]

? ? ? ??public string Name { get; set; }

? ? ? ??[DataMember]

? ? ? ??public int Age { get; set; }

? ? ? ??[DataMember]

? ? ? ??public string Sex { get; set; }

? ??}

}

當(dāng)執(zhí)行完上述反序列化操作后,如果反序列化成功,我們就可以得到一個還原后的Student 類對象,并且可以通過訪問對象的屬性來獲取相應(yīng)的值,例如像代碼中那樣輸出IDNameAgeSex 等屬性的值,以此來驗(yàn)證反序列化的結(jié)果是否符合預(yù)期哦,這樣就完成了從?JSON字符串到C#對象的反序列化過程啦。

四、借助?System.Text.Json進(jìn)行對象映射


類的準(zhǔn)備特點(diǎn)

在使用System.Text.Json 進(jìn)行對象映射相關(guān)操作時,其類的準(zhǔn)備與之前介紹的一些方式有所不同。它并不依賴像System.Runtime.Serialization.Json 中那樣特定的屬性標(biāo)記(如[DataContract] [DataMember] 等)來決定哪些屬性參與序列化和反序列化。

System.Text.Json 主要依據(jù)公共屬性和字段的可見性來進(jìn)行序列化。只要類中的屬性是公共的(使用public 修飾符),在默認(rèn)情況下就會被包含在序列化和反序列化的過程中。例如,我們定義如下一個簡單的Employee 類:

public class Employee

{

? ??public string Name { get; set; }

? ??public int Age { get; set; }

? ??public string Department { get; set; }

}

在上述代碼中,NameAge Department 這幾個公共屬性,在使用System.Text.Json 進(jìn)行序列化時,都會自動被處理,不需要額外添加特殊的標(biāo)記。這使得在一些簡單的場景下,類的準(zhǔn)備工作更加簡潔直觀,減少了代碼中額外的屬性標(biāo)記負(fù)擔(dān)。不過,如果想要對序列化和反序列化的過程進(jìn)行更精細(xì)的控制,比如忽略某些屬性或者改變屬性序列化后的名稱等,就需要借助System.Text.Json 提供的相關(guān)配置選項(xiàng)和特性來實(shí)現(xiàn)啦,后續(xù)我們會詳細(xì)講到哦。

序列化操作流程

下面來詳細(xì)說說利用System.Text.Json 進(jìn)行對象序列化的具體操作流程哈。

首先,我們要創(chuàng)建好需要序列化的對象,就像前面定義的Employee 類,我們可以這樣實(shí)例化一個對象:

Employee emp = new Employee

{

? ??Name = "小李",

? ??Age = 28,

? ??Department = "研發(fā)部"

};

接著,配置序列化相關(guān)的選項(xiàng),這一步是通過JsonSerializerOptions 類來完成的。這個類提供了很多可配置的屬性,例如:

?WriteIndented 屬性:如果將其設(shè)置為true,那么序列化后的?JSON字符串會進(jìn)行格式化,有縮進(jìn)和換行,方便閱讀查看;若設(shè)置為false(默認(rèn)值),則序列化后的?JSON字符串是緊湊的,沒有多余的空格和換行哦。示例代碼如下:

JsonSerializerOptions options = new JsonSerializerOptions

{

? ??WriteIndented = true

};

?PropertyNamingPolicy 屬性:可以用來指定屬性名稱在序列化后的命名規(guī)則,比如設(shè)置為JsonNamingPolicy.CamelCase,那么原本在?C#類中采用帕斯卡命名法(首字母大寫)的屬性名,在序列化后的JSON字符串中就會變成駝峰命名法(首字母小寫)形式啦。像這樣:

JsonSerializerOptions options = new JsonSerializerOptions

{

? ??PropertyNamingPolicy = JsonNamingPolicy.CamelCase

};

配置好選項(xiàng)后,就可以執(zhí)行序列化操作了,調(diào)用JsonSerializer.Serialize 方法,將對象和配置選項(xiàng)(如果有配置的話)作為參數(shù)傳入,示例代碼如下:

string jsonString = JsonSerializer.Serialize(emp, options);

Console.WriteLine(jsonString);

假設(shè)我們前面創(chuàng)建的Employee 對象屬性值分別是Name = "小李"Age = 28Department = "研發(fā)部",并且配置了WriteIndented = true PropertyNamingPolicy = JsonNamingPolicy.CamelCase,那最終序列化得到的?JSON字符串結(jié)果可能類似這樣:

{

? ??"name": "小李",

? ??"age": 28,

? ??"department": "研發(fā)部"

}

這樣就完成了利用System.Text.Json ?C#對象序列化為JSON字符串的操作啦,整個過程還是比較清晰明了的哦。

反序列化操作流程

接下來講講如何通過System.Text.Json ?JSON字符串反序列化為C#對象呀。

第一步,我們得先有一個?JSON字符串,這個字符串可能是從外部接口獲取到的,或者是從文件中讀取出來的數(shù)據(jù)等等,比如下面這樣一個示例JSON字符串:

{"name": "小張", "age": 25, "department": "市場部"}

?C#代碼中可以這樣定義這個字符串變量:

string json = "{\"name\": \"小張\", \"age\": 25, \"department\": \"市場部\"}";

然后,同樣需要配置反序列化相關(guān)的選項(xiàng),也是通過JsonSerializerOptions 類來進(jìn)行配置哦,配置的方法和序列化時類似,例如設(shè)置屬性名稱大小寫不敏感等情況(如果有需要的話),像這樣:

JsonSerializerOptions options = new JsonSerializerOptions

{

? ??PropertyNameCaseInsensitive = true

};

接著,就可以調(diào)用JsonSerializer.Deserialize 方法來執(zhí)行反序列化操作啦,需要指定要反序列化的目標(biāo)對象類型,示例代碼如下:

Employee deserializedEmp = JsonSerializer.Deserialize<Employee>(json, options);

如果反序列化成功,我們就可以通過訪問deserializedEmp 對象的各個屬性來獲取相應(yīng)的值啦,比如輸出對象的屬性值看看:

Console.WriteLine($"Name: {deserializedEmp.Name}, Age: {deserializedEmp.Age}, Department: {deserializedEmp.Department}");

這樣,就完成了從?JSON字符串到C#對象的反序列化過程哦,通過System.Text.Json 進(jìn)行序列化和反序列化操作,在很多項(xiàng)目中都能方便地實(shí)現(xiàn)數(shù)據(jù)的轉(zhuǎn)換和處理呢。

五、使用?Newtonsoft.JsonJson.NET)完成對象映射

程序包引用與命名空間導(dǎo)入

?C#項(xiàng)目中若要使用Newtonsoft.JsonJson.NET)庫來進(jìn)行?JSON序列化/反序列化操作,首先需要引用相應(yīng)的程序包并導(dǎo)入對應(yīng)的命名空間哦。

引用程序包的常見方式是通過?NuGet程序包管理器,不同的開發(fā)工具操作略有不同。例如在Visual Studio中,可以點(diǎn)擊工具- NuGet包管理器-程序包管理器控制臺,然后在控制臺輸入命令“Install-Package Newtonsoft.Json”(不同版本可能命令稍有差異,比如有的版本后面會跟具體版本號等),回車后NuGet就會自動下載并安裝Newtonsoft.Json程序包到項(xiàng)目中啦,像這樣就引用成功了哦。

在成功引用程序包后,我們還需要在代碼文件中導(dǎo)入對應(yīng)的命名空間,這樣才能使用該庫提供的各種類和方法呀,導(dǎo)入的語句是“using Newtonsoft.Json;”,將這條語句添加到代碼文件的頂部(通常在其他using語句一起的位置哦),就可以開始使用Newtonsoft.Json相關(guān)功能進(jìn)行對象映射的操作啦,下面咱們接著看看具體的序列化和反序列化示例吧。

序列化示例與特性運(yùn)用

接下來看看如何利用?Newtonsoft.Json中的JsonConvert.SerializeObject方法將C#對象序列化為JSON字符串哈。

咱們先定義一個簡單的類,比如下面這個表示學(xué)生信息的類:

public class Student

{

? ??public int Id { get; set; }

? ??public string Name { get; set; }

? ??public string Sex { get; set; }

? ??public string Description { get; set; }

}

然后創(chuàng)建這個類的對象,并調(diào)用?JsonConvert.SerializeObject方法進(jìn)行序列化操作,示例代碼如下:

List<Student> students = new List<Student>();

students.Add(new Student { Id = 1, Name = "張三", Sex = "", Description = "班長" });

students.Add(new Student { Id = 2, Name = "李四", Sex = "", Description = "小組長" });

students.Add(new Student { Id = 3, Name = "王五", Sex = "", Description = "宣傳委員" });

string studentsJson = JsonConvert.SerializeObject(students);

Console.WriteLine(studentsJson);

上述代碼運(yùn)行后,輸出結(jié)果會是這樣的?JSON字符串格式:

[{"Id":1,"Name":"張三","Sex":"","Description":"班長"},{"Id":2,"Name":"李四","Sex":"","Description":"小組長"},{"Id":3,"Name":"王五","Sex":"","Description":"宣傳委員"}]

在實(shí)際應(yīng)用中,有時候我們可能不希望某些字段參與對象的?JSON序列化,這時候就可以使用[JsonIgnore]特性啦。例如,我們不想讓“Sex”字段被序列化,那可以修改“Student”類的定義如下:

public class Student

{

? ??public int Id { get; set; }

? ??public string Name { get; set; }

? ??[JsonIgnore]

? ??public string Sex { get; set; }

? ??public string Description { get; set; }

}

再執(zhí)行同樣的序列化操作,輸出結(jié)果就會變成這樣:

[{"Id":1,"Name":"張三","Description":"班長"},{"Id":2,"Name":"李四","Description":"小組長"},{"Id":3,"Name":"王五","Description":"宣傳委員"}]

可以看到“Sex”字段就沒有出現(xiàn)在序列化后的JSON字符串中了哦。

還有一種情況,我們可能想在對象序列化時更換字段的輸出名稱,比如簡化字段名稱以縮短?JSON字符串的長度等,這時就可以使用[JsonProperty]特性標(biāo)識字段啦。假設(shè)我們想把“Description”字段在序列化后的名稱改為“Desc”,代碼可以這樣寫:

public class Student

{

? ??public int Id { get; set; }

? ??public string Name { get; set; }

? ??[JsonIgnore]

? ??public string Sex { get; set; }

? ??[JsonProperty("Desc")]

? ??public string Description { get; set; }

}

重新執(zhí)行序列化操作后,輸出結(jié)果如下:

[{"Id":1,"Name":"張三","Desc":"班長"},{"Id":2,"Name":"李四","Desc":"小組長"},{"Id":3,"Name":"王五","Desc":"宣傳委員"}]

通過這些特性的運(yùn)用,我們可以更靈活地控制對象序列化后的?JSON字符串格式呢,方便滿足不同的業(yè)務(wù)需求哦。

反序列化示例與注意事項(xiàng)

下面咱們講講如何將?JSON字符串通過JsonConvert.DeserializeObject方法反序列化為C#對象哈。

先來看個簡單的例子,假設(shè)我們有如下的?JSON字符串:

[{"Id":1, "Name": "張三", "Sex": "", "Description": "班長"},{"Id":2, "Name": "李四", "Sex": "", "Description": "小組長"},{"Id":2, "Name": "王五", "Sex": "", "Description": "宣傳委員"}]

我們要把它反序列化為“Student”類的對象列表,對應(yīng)的“Student”類定義如下(這里注意不要使用之前講序列化特性時那些特殊的特性標(biāo)記哦,保持常規(guī)的類定義就行):

public class Student

{

? ??public int Id { get; set; }

? ??public string Name { get; set; }

? ??public string Sex { get; set; }

? ??public string Description { get; set; }

}

?C#代碼中進(jìn)行反序列化的操作示例如下:

string inputJsonString = @"[{"Id":1, "Name": "張三", "Sex": "", "Description": "班長"},{"Id":2, "Name": "李四", "Sex": "", "Description": "小組長"},{"Id":2, "Name": "王五", "Sex": "", "Description": "宣傳委員"}]";

List<Student> objects = JsonConvert.DeserializeObject<List<Student>>(inputJsonString);

foreach (Student item in objects)

{

? ??Console.WriteLine($"Id: {item.Id}, Name: {item.Name}, Sex: {item.Sex}, Description: {item.Description}");

}

上述代碼執(zhí)行后,就能成功將?JSON字符串反序列化為“Student”類的對象列表,并且可以通過遍歷輸出每個對象的屬性值來驗(yàn)證反序列化的結(jié)果哦,輸出結(jié)果會是這樣:

Id: 1, Name:張三, Sex:, Description:班長

Id: 2, Name:李四, Sex:, Description:小組長

Id: 2, Name:王五, Sex:, Description:宣傳委員

這里有個需要注意的事項(xiàng)哈,在進(jìn)行反序列化時,類中屬性名稱要和?JSON字符串中的鍵相對應(yīng)哦。如果屬性名稱不一致,那可能會導(dǎo)致反序列化出來的對象屬性值不正確或者出現(xiàn)異常情況呢。比如JSON字符串里某個鍵是“student_name”,而對應(yīng)的C#類里屬性定義是“StudentName”,那就沒辦法正確賦值啦,所以在編寫代碼和設(shè)計數(shù)據(jù)結(jié)構(gòu)時,要保證這種對應(yīng)關(guān)系哦,這樣才能順利地完成基于Newtonsoft.Json的對象反序列化操作呀,大家可以自己動手多試試不同的示例,加深理解哦。

六、不同方法的對比與選擇建議

各方法性能、功能對比

?C#開發(fā)中,常用的JSON序列化/反序列化方法有System.Runtime.Serialization.JsonSystem.Text.JsonNewtonsoft.Json等,它們在性能表現(xiàn)以及支持的功能特性方面存在一定差異。

?System.Runtime.Serialization.Json:這是.NET Framework內(nèi)置的庫,優(yōu)勢在于無需額外的引用,并且支持DataContract屬性和WCF協(xié)定,比較適合在一些老項(xiàng)目中使用。不過,其性能相對較差,像在處理大量數(shù)據(jù)的序列化和反序列化時,效率可能不如其他幾種方式。而且它不支持一些高級功能,例如自定義轉(zhuǎn)換器,在面對復(fù)雜的JSON數(shù)據(jù)場景時,靈活性欠佳。

?System.Text.Json:作為.NET Core的一部分,在創(chuàng)建跨平臺應(yīng)用程序時,不需要額外的依賴項(xiàng),使用起來較為方便。性能方面進(jìn)行了優(yōu)化,通常比Newtonsoft.Json更快,利用了新的讀寫API,采用更高效的內(nèi)部實(shí)現(xiàn),有更好的性能和內(nèi)存利用率。在處理簡單的JSON數(shù)據(jù)時,其提供的簡化API也更容易操作,但默認(rèn)是強(qiáng)類型轉(zhuǎn)換,比如實(shí)體類定義的是字符串,JSON字符串返回的是整型,轉(zhuǎn)換時會報錯。另外,雖然默認(rèn)強(qiáng)類型轉(zhuǎn)換,但也可以通過自定義轉(zhuǎn)換器等方式來實(shí)現(xiàn)弱類型轉(zhuǎn)換,以滿足更多不同的需求。

?Newtonsoft.JsonJSON.NET:它是一個成熟且使用廣泛的第三方庫,在.NET社區(qū)中被廣泛接受和使用。擁有豐富的功能和強(qiáng)大的靈活性,比如支持完全自定義的序列化和反序列化邏輯,能處理循環(huán)引用、自定義轉(zhuǎn)換器、忽略屬性等等,可以方便地應(yīng)對各種復(fù)雜的JSON數(shù)據(jù)場景。不過,它相對有較大的依賴包,并且在某些場景下,性能可能會低于一些最新的庫。

根據(jù)項(xiàng)目場景選擇合適方法

在實(shí)際的項(xiàng)目開發(fā)中,選擇哪種?JSON序列化/反序列化方法來實(shí)現(xiàn)對象映射,需要綜合考慮多個因素,以下是一些依據(jù)不同項(xiàng)目情況給出的選擇建議:

?基于.NET版本

?如果是老項(xiàng)目,本身基于.NET Framework,且對性能要求不是特別極致,功能需求也相對常規(guī),那么System.Runtime.Serialization.Json是可以勝任的,畢竟無需額外引用,能方便地集成到已有項(xiàng)目中,符合項(xiàng)目原本的技術(shù)棧特點(diǎn)。

?若是基于.NET Core開發(fā)的新項(xiàng)目,特別是創(chuàng)建跨平臺應(yīng)用程序時,System.Text.Json就比較合適了,它與.NET Core集成緊密,還自帶性能優(yōu)勢以及簡化的API,方便快捷地處理常見的JSON數(shù)據(jù)操作,減少額外依賴帶來的潛在問題。

?而無論項(xiàng)目基于哪種.NET版本,只要涉及到較為復(fù)雜的JSON數(shù)據(jù)處理場景,比如需要處理循環(huán)引用、要對序列化和反序列化邏輯做深度定制、或者要靈活地處理各種數(shù)據(jù)類型轉(zhuǎn)換等情況,Newtonsoft.Json就是一個很好的選擇,雖然可能依賴包稍大些,但豐富的功能可以應(yīng)對各種復(fù)雜需求。

?對性能和靈活性的需求

?當(dāng)項(xiàng)目對性能有較高要求,例如是大數(shù)據(jù)量處理的應(yīng)用,像電商平臺的訂單數(shù)據(jù)處理、大數(shù)據(jù)系統(tǒng)中的數(shù)據(jù)交互等場景,優(yōu)先考慮?System.Text.Json,它在處理大量數(shù)據(jù)的序列化和反序列化時效率表現(xiàn)突出,能夠保障系統(tǒng)整體的數(shù)據(jù)處理速度,避免性能瓶頸。

?如果項(xiàng)目更注重靈活性,比如需要頻繁根據(jù)不同業(yè)務(wù)場景去動態(tài)調(diào)整對象序列化和反序列化的規(guī)則,像根據(jù)不同客戶端需求輸出不同格式的?JSON數(shù)據(jù)、要忽略某些屬性或者改變屬性序列化后的名稱等情況,Newtonsoft.Json的優(yōu)勢就體現(xiàn)出來了,借助其豐富的功能特性,可以輕松實(shí)現(xiàn)各種自定義邏輯,滿足多變的業(yè)務(wù)需求。

?要是項(xiàng)目只是處理一些簡單常規(guī)的?JSON數(shù)據(jù)交互,性能要求也不是特別高,追求簡潔方便的開發(fā)體驗(yàn),System.Text.Json同樣可以滿足,利用它默認(rèn)的配置和簡單的API就能快速實(shí)現(xiàn)對象映射,減少開發(fā)成本和出錯概率。

七、總結(jié)

內(nèi)容回顧

在本文中,我們詳細(xì)介紹了利用?JSON序列化/反序列化實(shí)現(xiàn)C#對象映射的幾種主要方法,下面來簡單回顧一下關(guān)鍵內(nèi)容,幫助大家強(qiáng)化記憶哦。

首先是利用System.Runtime.Serialization.Json 實(shí)現(xiàn)對象映射,使用這個命名空間時,參與操作的類要標(biāo)記[DataContract],類中的屬性需標(biāo)記[DataMember],這是進(jìn)行序列化和反序列化的基礎(chǔ)準(zhǔn)備。序列化時,要先創(chuàng)建對象、實(shí)例化DataContractJsonSerializer 序列化器、創(chuàng)建MemoryStream 內(nèi)存流對象,接著用序列化器的WriteObject 方法將對象寫入內(nèi)存流,最后從內(nèi)存流中讀取數(shù)據(jù)并轉(zhuǎn)換為字符串,就得到了序列化后的?JSON字符串。反序列化與之類似,先有JSON字符串,實(shí)例化反序列化器后,用ReadObject 方法從內(nèi)存流中讀取數(shù)據(jù)還原成?C#對象。

System.Text.Json 進(jìn)行對象映射時,類準(zhǔn)備工作主要依據(jù)公共屬性和字段的可見性,公共屬性默認(rèn)會被包含在序列化和反序列化過程中。序列化要先創(chuàng)建對象,通過JsonSerializerOptions 類配置相關(guān)選項(xiàng),比如設(shè)置WriteIndented 屬性控制字符串格式、PropertyNamingPolicy 屬性指定屬性名稱的命名規(guī)則等,然后調(diào)用JsonSerializer.Serialize 方法執(zhí)行序列化。反序列化則是先有?JSON字符串,同樣配置好JsonSerializerOptions 類(按需配置屬性名大小寫不敏感等情況),再調(diào)用JsonSerializer.Deserialize 方法還原?C#對象。

使用Newtonsoft.JsonJson.NET 庫時,得先通過?NuGet程序包管理器引用程序包,并在代碼文件中導(dǎo)入using Newtonsoft.Json; 命名空間。序列化可以通過JsonConvert.SerializeObject 方法實(shí)現(xiàn),還能運(yùn)用特性,像[JsonIgnore] 可讓字段不參與序列化,[JsonProperty] 能更換字段的輸出名稱,使序列化更靈活。反序列化用JsonConvert.DeserializeObject 方法,不過要注意類中屬性名稱要和?JSON字符串中的鍵對應(yīng),不然可能出現(xiàn)賦值異常情況哦。

不同的方法各有特點(diǎn),在性能、功能以及適用場景等方面都存在差異,大家可以根據(jù)實(shí)際項(xiàng)目情況去選擇合適的方式來進(jìn)行?C#對象映射呀。

拓展與展望

隨著技術(shù)的不斷發(fā)展,JSON序列化/反序列化在C#對象映射中的應(yīng)用也有著廣闊的拓展空間呢。

從拓展應(yīng)用方向來看,在微服務(wù)架構(gòu)日益流行的當(dāng)下,各個微服務(wù)之間的數(shù)據(jù)交互頻繁,利用?JSON序列化/反序列化實(shí)現(xiàn)對象映射可以更好地保障數(shù)據(jù)格式的統(tǒng)一和交互的順暢。例如,在一個電商系統(tǒng)的微服務(wù)架構(gòu)里,商品服務(wù)、訂單服務(wù)、用戶服務(wù)等不同的微服務(wù)之間傳遞數(shù)據(jù)時,通過高效準(zhǔn)確的對象映射,能讓數(shù)據(jù)快速在各個環(huán)節(jié)流轉(zhuǎn),提升整個系統(tǒng)的協(xié)同效率。

而且,在跨平臺開發(fā)越來越普遍的趨勢下,像.NET Core支持的跨平臺應(yīng)用開發(fā)場景中,System.Text.Json 這類無需額外依賴項(xiàng)且性能優(yōu)良的方式,有望在更多類型的項(xiàng)目里發(fā)揮重要作用,進(jìn)一步優(yōu)化數(shù)據(jù)傳輸和存儲的效率。

對于想要進(jìn)一步學(xué)習(xí)提升的朋友們來說,可以深入研究各個庫的高級特性和自定義功能哦。比如深入了解Newtonsoft.Json 的自定義轉(zhuǎn)換器實(shí)現(xiàn)原理,嘗試自己編寫復(fù)雜數(shù)據(jù)結(jié)構(gòu)的轉(zhuǎn)換邏輯,應(yīng)對各種特殊的業(yè)務(wù)數(shù)據(jù)需求;或者探索System.Text.Json 如何更好地結(jié)合.NET Core的其他新特性,來打造高性能、低資源占用的應(yīng)用程序。

此外,多參考開源項(xiàng)目里對于?JSON序列化/反序列化對象映射的優(yōu)秀實(shí)踐案例也是個不錯的學(xué)習(xí)途徑,從中汲取經(jīng)驗(yàn),運(yùn)用到自己的實(shí)際項(xiàng)目中。


該文章在 2024/12/19 17:56:04 編輯過
關(guān)鍵字查詢
相關(guān)文章
正在查詢...
點(diǎn)晴ERP是一款針對中小制造業(yè)的專業(yè)生產(chǎn)管理軟件系統(tǒng),系統(tǒng)成熟度和易用性得到了國內(nèi)大量中小企業(yè)的青睞。
點(diǎn)晴PMS碼頭管理系統(tǒng)主要針對港口碼頭集裝箱與散貨日常運(yùn)作、調(diào)度、堆場、車隊、財務(wù)費(fèi)用、相關(guān)報表等業(yè)務(wù)管理,結(jié)合碼頭的業(yè)務(wù)特點(diǎn),圍繞調(diào)度、堆場作業(yè)而開發(fā)的。集技術(shù)的先進(jìn)性、管理的有效性于一體,是物流碼頭及其他港口類企業(yè)的高效ERP管理信息系統(tǒng)。
點(diǎn)晴WMS倉儲管理系統(tǒng)提供了貨物產(chǎn)品管理,銷售管理,采購管理,倉儲管理,倉庫管理,保質(zhì)期管理,貨位管理,庫位管理,生產(chǎn)管理,WMS管理系統(tǒng),標(biāo)簽打印,條形碼,二維碼管理,批號管理軟件。
點(diǎn)晴免費(fèi)OA是一款軟件和通用服務(wù)都免費(fèi),不限功能、不限時間、不限用戶的免費(fèi)OA協(xié)同辦公管理系統(tǒng)。
Copyright 2010-2025 ClickSun All Rights Reserved