JSON
JSON(JavaScript Object Notation) 是一種輕量級(jí)的數(shù)據(jù)交換格式个粱。它基于 [ECMAScript] (歐洲計(jì)算機(jī)協(xié)會(huì)制定的js規(guī)范)的一個(gè)子集,采用完全獨(dú)立于編程語(yǔ)言的文本格式來(lái)存儲(chǔ)和表示數(shù)據(jù)鬼悠。簡(jiǎn)潔和清晰的層次結(jié)構(gòu)使得 JSON 成為理想的數(shù)據(jù)交換語(yǔ)言魄揉。 易于人閱讀和編寫凄鼻,同時(shí)也易于機(jī)器解析和生成肢础,并有效地提升網(wǎng)絡(luò)傳輸效率。
JSON 語(yǔ)法規(guī)則
在 JS 語(yǔ)言中肴掷,一切都是對(duì)象敬锐。因此,任何支持的類型都可以通過 JSON 來(lái)表示呆瞻,例如字符串台夺、數(shù)字、對(duì)象痴脾、數(shù)組等颤介。但是對(duì)象和數(shù)組是比較特殊且常用的兩種類型:
對(duì)象表示為鍵值對(duì)
數(shù)據(jù)由逗號(hào)分隔
花括號(hào)保存對(duì)象
方括號(hào)保存數(shù)組
JSON 鍵/值對(duì)
JSON 鍵值對(duì)是用來(lái)保存 JS 對(duì)象的一種方式,和 JS 對(duì)象的寫法也大同小異赞赖,鍵/值對(duì)組合中的鍵名寫在前面并用雙引號(hào) "" 包裹滚朵,使用冒號(hào) : 分隔,然后緊接著值:
{"firstName": "Json"}
這很容易理解前域,等價(jià)于這條 JavaScript 語(yǔ)句:
{firstName : "Json"}
JSON 與 JS 對(duì)象的關(guān)系
很多人搞不清楚 JSON 和 Js 對(duì)象的關(guān)系始绍,甚至連誰(shuí)是誰(shuí)都不清楚。其實(shí)话侄,可以這么理解:
JSON 是 JS 對(duì)象的字符串表示法,它使用文本表示一個(gè) JS 對(duì)象的信息,本質(zhì)是一個(gè)字符串年堆。
如
var obj = {a: 'Hello', b: 'World'};
//這是一個(gè)對(duì)象吞杭,注意鍵名也是可以使用引號(hào)包裹的。
var json = '{"a": "Hello", "b": "World"}';
//這是一個(gè) JSON 字符串变丧,本質(zhì)是一個(gè)字符串芽狗。
和XML的比較
可讀性
JSON和XML的可讀性可謂不相上下,一邊是簡(jiǎn)易的語(yǔ)法痒蓬,一邊是規(guī)范的標(biāo)簽形式童擎,很難分出勝負(fù)。
可擴(kuò)展性
XML天生有很好的擴(kuò)展性攻晒,JSON當(dāng)然也有顾复,沒有什么是XML可以擴(kuò)展而JSON卻不能擴(kuò)展的。而且JSON可以存儲(chǔ)Javascript復(fù)合對(duì)象鲁捏,有著xml不可比擬的優(yōu)勢(shì)芯砸。
Json解析-LitJson
using LitJson;
using System;
public class Person
{
// C# 3.0 auto-implemented properties
public string Name { get; set; }
public int Age { get; set; }
public DateTime Birthday { get; set; }
}
public class JsonSample
{
public static void Main()
{
PersonToJson();
JsonToPerson();
}
public static void PersonToJson()
{
Person bill = new Person();
bill.Name = "William Shakespeare";
bill.Age = 51;
bill.Birthday = new DateTime(1564, 4, 26);
string json_bill = JsonMapper.ToJson(bill);
Console.WriteLine(json_bill);
}
public static void JsonToPerson()
{
string json = @"
{
""Name"" : ""Thomas More"",
""Age"" : 57,
""Birthday"" : ""02/07/1478 00:00:00""
}";
Person thomas = JsonMapper.ToObject<Person>(json);
Console.WriteLine("Thomas' age: {0}", thomas.Age);
}
}
Json解析-JsonUtility
JsonUtility-Unity5引入的Json工具,
原本Json相對(duì)簡(jiǎn)單给梅,但是這套工具使用的是Unity Serializer假丧。和我們?nèi)粘J褂玫腏son序列化有點(diǎn)差異。
支持序列化MonoBehaviour的子類动羽,ScriptableObject的子類以及包含[Serializable]標(biāo)簽的類和結(jié)構(gòu)包帚。
這套Api主要優(yōu)勢(shì)是Unity內(nèi)置,可以序列化和反序列化Vector3等數(shù)據(jù)运吓。在很多簡(jiǎn)單的場(chǎng)景渴邦,可以方便我們開發(fā)∮鸬拢總的來(lái)說就是不好用几莽。可以結(jié)合其他json工具使用宅静。
序列化ScriptableObject和MonoBehavior
public class JsonTest : MonoBehaviour
{
/// <summary>
/// 序列化字段
/// </summary>
public int Value;
void Start()
{
// 賦值Value為2
Value = 1;
// {"Value":1}
var json = JsonUtility.ToJson(this);
// 賦值Value為2
Value = 2;
// 數(shù)據(jù)被重新寫入章蚣,Value賦值為1
JsonUtility.FromJsonOverwrite(json, this);
// ArgumentException: Cannot deserialize JSON to new instances of type 'JsonTest.'
// 不支持直接反序列化生成一個(gè)MonoBehavior的子類
var fromJson = JsonUtility.FromJson<JsonTest>(json);
}
}
序列化List和Dictionary和Array
如果要支持List和Array,建議寫一個(gè)包裝類姨夹,將其包含再里邊纤垂。并且泛型中加入[Serializable]標(biāo)簽
如果要支持Dictionary,建議改寫成List磷账。
// 沒有包含[Serializable]的類峭沦,不可被包含在序列化類內(nèi)
public class Item1
{
public string Str;
}
// 包含標(biāo)簽可以正常序列化
[Serializable]
public class Item2
{
public string Str;
}
public class ListCollection
{
// 可正常序列化
public List<int> Item0;
// 沒有標(biāo)簽不可正常序列化
public List<Item1> Item1;
// 有標(biāo)簽可以正常序列化
public List<Item2> Item2;
}
public void Run()
{
var collection = new ListCollection()
{
Item0 = new List<int>() { 0, 1, 2, 3, 4 },
Item1 = new List<Item1>() { new Item1() { Str = "1" }, new Item1() { Str = "2" } },
Item2 = new List<Item2>() { new Item2() { Str = "1" }, new Item2() { Str = "2" } }
};
// {"Item0":[0,1,2,3,4],"Item2":[{"Str":"1"},{"Str":"2"}]}
Debug.Log(JsonUtility.ToJson(collection));
// {}
Debug.Log(JsonUtility.ToJson(collection.Item0));
// {}
Debug.Log(JsonUtility.ToJson(collection.Item1));
// {}
Debug.Log(JsonUtility.ToJson(collection.Item2));
// ArgumentException: JSON must represent an object type.
var fromJson = JsonUtility.FromJson<List<int>>("[0,1,2,3,4]");
Debug.Log(fromJson.Count);
}
這里提供一個(gè)快速包裝方法。需要的朋友可以考慮使用逃糟。
// 出處: https://forum.unity3d.com/threads/how-to-load-an-array-with-jsonutility.375735/#post-2585129
public class JsonHelper
{
public static T[] getJsonArray<T>(string json)
{
string newJson = "{ \"array\": " + json + "}";
Wrapper<T> wrapper = JsonUtility.FromJson<Wrapper<T>> (newJson);
return wrapper.array;
}
[Serializable]
private class Wrapper<T>
{
public T[] array;
}
}