JSON
AJAX传递复杂数据如果自己进行格式定义的话会经历组装、解析的过程,因此AJAX中有一个事实上的数据传输标准JSon。Json(是一个标准,就像XML一样,Json规定了对象以什么样的格式保存为一个字符串)将复杂对象序列化为一个字符串,在浏览器端再将字符串反序列化为JavaScript可以读取的对象。看一下Json的格式。Json被几乎所有语言支持。
C#中将.Net对象序列化为Json字符串的方法:JavaScriptSerializer().Serialize§, JavaScriptSerializer在System.Web.Extensions.dll中,是.Net3.x 中新增的类。完整:System.Web.Script.Serialization.JavaScriptSerializer
一、什么是JSON
JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式。它基于JavaScript(Standard ECMA-262 3rd Edition - December 1999)的一个子集。 JSON采用完全独立于语言的文本格式,但是也使用了类似于C语言家族的习惯(包括C, C++, C#, Java, JavaScript, Perl, Python等)。这些特性使JSON成为理想的数据交换语言。易于人阅读和编写,同时也易于机器解析和生成。
二、和XML的比较
可读性:JSON和XML的可读性可谓不相上下,一边是简易的语法,一边是规范的标签形式,很难分出胜负。
可扩展性:XML天生有很好的扩展性,JSON当然也有,没有什么是XML能扩展,而JSON却不能扩展的。不过JSON在Javascript主场作战,可以存储Javascript复合对象,有着xml不可比拟的优势。
编码难度:XML有丰富的编码工具,比如Dom4j、JDom等,JSON也有提供的工具。无工具的情况下,相信熟练的开发人员一样能很快的写出想要的xml文档和JSON字符串,不过,xml文档要多很多结构上的字符。
三、json格式
1、对象是一个无序的“‘名称/值’对”集合。
例子:表示人的一个对象:
{“姓名” : “大憨”,“年龄” : 24}
2、数组是值(value)的有序集合。
例子:一组学生
{“学生” :[{“姓名” : “小明” , “年龄” : 23},{“姓名” : “大憨” , “年龄” : 24}]}
json 对象数组:[{“姓名” : “小明” , “年龄” : 23},{“姓名” : “大憨” , “年龄” : 24}]
3、值(value)可以是双引号括起来的字符串(string)、数值(number)、true、false、 null、对象(object)或者数组(array)。这些结构可以嵌套。
4、字符串(string)是由双引号包围的任意数量Unicode字符的集合,使用反斜线转义。一个字符(character)即一个单独的字符串(character string)。 字符串(string)与C或者Java的字符串非常相似。
5、数值(number)也与C或者Java的数值非常相似。除去未曾使用的八进制与十六进制格式。除去一些编码细节。
四、C#调用
主要使用类 | 需要的命名空间 | 限制 | 是否支持linq | |
---|---|---|---|---|
1 | DataContractJsonSerializer | System.Runtime.Serialization.Json | 通用 | 否 |
2 | JavaScriptSerializer | System.Web.Script.Serialization | 常用于web中 | 否 |
3 | JsonArray、JsonObject、JsonValue | System.Json | 常用于Silverlight中 | 是 |
4 | JsonConvert、JArray、JObject、JValue、JProperty | Newtonsoft.Json | 通用 | 是 |
五、详细使用
1.契约方式,使用DataContractJsonSerializer
1.1 需要添加引用,并引入命名空间:using System.Runtime.Serialization;
1.2 准备数据:
[DataContract]
public class Person
{
[DataMember(Order = 0, IsRequired = true)]
public string Name { get; set; }
[DataMember(Order = 1)]
public int Age { get; set; }
[DataMember(Order = 2)]
public bool Alive { get; set; }
[DataMember(Order = 3)]
public string[] FavoriteFilms { get; set; }
[DataMember(Order = 4)]
public Person Child { get; set; }
}
1.3 测试帮助类
public static class JSON
{
public static T parse<T>(string jsonString)
{
using (var ms = new MemoryStream(Encoding.UTF8.GetBytes(jsonString)))
{
return (T)new DataContractJsonSerializer(typeof(T)).ReadObject(ms);
}
}
public static string stringify(object jsonObject)
{
using (var ms = new MemoryStream())
{
new DataContractJsonSerializer(jsonObject.GetType()).WriteObject(ms, jsonObject);
return Encoding.UTF8.GetString(ms.ToArray());
}
}
}
1.4 main函数调用
//类初始化
var p1 = new Person
{
Age = 10,
Alive = true,
Name = "yylp521",
FavoriteFilms = new[] { "行尸走肉", "生化危机" }
};
var p2 = new Person() { Age = 32, Name = "bjlihx", Child = p1 };
//序列化
var jsonString = JSON.stringify(new[] { p1, p2 });
Console.WriteLine(jsonString);
JSON.parse<List<Person>>(jsonString);// 反序列化,泛型集合
JSON.parse<Person[]>(jsonString);// 数组转换
Console.ReadKey();
输出数据:[{“Name”:“yylp521”,“Age”:10,“Alive”:true,“FavoriteFilms”:[“行尸走肉”,“生化危机”],“Child”:null},{“Name”:“bjlihx”,“Age”:32,“Alive”:false,“FavoriteFilms”:null,“Child”:{“Name”:“yylp521”,“Age”:10,“Alive”:true,“FavoriteFilms”:[“行尸走肉”,“生化危机”],“Child”:null}}]
简要说明:
1.准备数据那里要有相关特性标签
2.也可以通过JsonReaderWriterFactory来实现
2.内置方式,使用JavaScriptSerializer
2.1 使用.NET Framework 3.5之后版本(包含.Net FrameWork 3.5)中提供的System.Web.Script.Serialization命名空间下的JavaScriptSerializer类进行对象的序列化与反序列化.
2.2 需要添加命名空间:using System.Web.Script.Serialization;
2.3 测试代码
JavaScriptSerializer json = new JavaScriptSerializer();
json.Serialize(new List<Person>() { p1, p2 });//这里参数是Object类型的
注:在.NET 2.0中没有内置序列化JSON的类,原因估计是当时Ajax尚未兴起。后来就有人写了一个json.net类库。.NET 3.5新增了一个把对象序列化为JSON字符串的类JavaScriptSerializer。这个类位于System.Web.Script.Serialization名字空间中(非Web项目需要添加System.Web.Extensions.dll引用).
3.支持Silverlight使用,system.json
4.通用开源方式,使用JSON.NET
4.1 使用开源的类库Newtonsoft.Json(下载地址http://json.codeplex.com/)。下载后加入工程就能用。
通常可以使用JObject, JsonReader, JsonWriter处理。这种方式最通用,最灵活,也可以随时修改。
4.2 详细内容
4.2.0 添加命名空间:using Newtonsoft.Json;
4.2.1 使用JsonReader读Json字符串
string jsonText = @"{""input"" : ""value"", ""output"" : ""result""}";
JsonReader reader = new JsonTextReader(new StringReader(jsonText));
while (reader.Read())
{
Console.WriteLine(reader.TokenType + "\t\t" + reader.ValueType + "\t\t" + reader.Value+"\r\n");
}
Console.ReadKey();
4.2.2 使用JsonWriter写字符串
StringWriter sw = new StringWriter();
JsonWriter writer = new JsonTextWriter(sw);
writer.WriteStartObject();
writer.WritePropertyName("input");
writer.WriteValue("value");
writer.WritePropertyName("output");
writer.WriteValue("result");
writer.WriteEndObject();
writer.Flush();
string jsonText2 = sw.GetStringBuilder().ToString();
Console.WriteLine(jsonText2);
Console.ReadKey();
4.2.3 使用JObject读写字符串
4.2.3.1 添加命名空间:using Newtonsoft.Json.Linq;
JObject jo = JObject.Parse(jsonText);
string[] values = jo.Properties().Select(item => item.Value.ToString()).ToArray();
4.2.4 使用JsonSerializer读写对象(基于JsonWriter与JsonReader)
4.2.4.1数组型数据
string jsonArrayText1 = "[{'a':'a1','b':'b1'},{'a':'a2','b':'b2'}]";
JArray ja = (JArray)JsonConvert.DeserializeObject(jsonArrayText1);
string ja1a = ja[1]["a"].ToString();
//或者
JObject o = (JObject)ja[1];
string oa = o["a"].ToString();
4.2.4.2嵌套格式
string jsonText3 = "{\"beijing\":{\"zone\":\"海淀\",\"zone_en\":\"haidian\"}}";
JObject jo1 = (JObject)JsonConvert.DeserializeObject(jsonText3);
string zone = jo1["beijing"]["zone"].ToString();
string zone_en = jo1["beijing"]["zone_en"].ToString();
4.2.4.3 自定义使用
1.辅助类
class Project
{
public string Input { get; set; }
public string Output { get; set; }
}
2.使用
Project p = new Project() { Input = "stone", Output = "gold" };
JsonSerializer serializer = new JsonSerializer();
StringWriter sw1 = new StringWriter();
serializer.Serialize(new JsonTextWriter(sw), p);
Console.WriteLine(sw.GetStringBuilder().ToString());
StringReader sr = new StringReader(@"{""Input"":""stone"", ""Output"":""gold""}");
Project p1 = (Project)serializer.Deserialize(new JsonTextReader(sr), typeof(Project));
Console.WriteLine(p1.Input + "=>" + p1.Output);
Console.ReadKey();
处理Json数据中的日期类型.如/Date(1415169703000)/格式
在asp.net mvc后台返回到视图中的json数据中想对数据进行操作,发现日期类型无法直接进行操作,需要转换为指定格式才行.在网上也搜了下方法也很多,觉得有点麻烦,最终使用正则搞定了,分享下:
var jsondate="/Date(1415169703000)/";
var formatdate=eval(jsondate.replace(/\/Date\((\d+)\)\//gi, "new Date($1)"));
alert(formatdate.toLocaleDateString());
有时取得的日期可能为这种类型"/Date(1415169703000+0800)/",这时代码就需要调整下了:
var jsondate="/Date(1415169703000+8000)/";
var formatdate=eval(jsondate.replace(/\/Date\((\d+)([\+\-](\d\d)(\d\d))?\)\//gi, "new Date($1)"));
alert(formatdate.toLocaleDateString());
运行结果如下: