序列化和反序列化的几种方式(JavaScriptSerializer 、XmlSerializer、DataContractSerializer)(一)
JavaScriptSerializer 类
为启用 AJAX 的应用程序提供序列化和反序列化功能。
命名空间: System.Web.Script.Serialization
程序集: System.Web.Extensions(在 System.Web.Extensions.dll 中)
JavaScriptSerializer 类由异步通信层内部使用,用于序列化和反序列化在浏览器和 Web 服务器之间传递的数据。 您无法访问序列化程序的此实例。 但是,此类公开了公共 API。 因此,当您希望在托管代码中使用 JavaScript 对象符号 (JSON) 时可以使用此类。
若要序列化对象,请使用 Serialize 方法。 若要反序列化 JSON 字符串,请使用 Deserialize 或 DeserializeObject 方法。 若要序列化和反序列化 JavaScriptSerializer 本身不支持的类型,请使用 JavaScriptConverter 类来实现自定义转换器。 然后,使用 RegisterConverters 方法注册转换器。
1 using System; 2 using System.Collections.Generic; 3 using System.Linq; 4 using System.Text; 5 using System.Threading.Tasks; 6 using System.Web.Script.Serialization; 7 namespace Wolfy.SerializerDemo 8 { 9 class Program 10 { 11 static void Main(string[] args) 12 { 13 List<Person> persons = new List<Person>{ 14 new Person("wolfy",24,Convert.ToDateTime("1989-04-11")), 15 new Person("张三",23,Convert.ToDateTime("1990-04-11")), 16 new Person("李四",22,Convert.ToDateTime("1991-04-11")), 17 new Person("王五",21,Convert.ToDateTime("1992-04-11")) 18 }; 19 //创建JavaScriptSerializer对象 20 JavaScriptSerializer jss = new JavaScriptSerializer(); 21 //调用序列化方法Serialize 22 string json = jss.Serialize(persons); 23 Console.WriteLine("序列化....."); 24 Console.WriteLine(json); 25 Console.WriteLine("反序列化....."); 26 JavaScriptSerializer jss2 = new JavaScriptSerializer(); 27 List<Person> list = jss2.Deserialize<List<Person>>(json); 28 foreach (Person item in list) 29 { 30 Console.WriteLine(item.Name + "\t" + item.Age.ToString() + "\t" + item.Birthday); 31 } 32 33 Console.Read(); 34 } 35 } 36 }
Person类
1 using System; 2 using System.Collections.Generic; 3 using System.Linq; 4 using System.Text; 5 using System.Threading.Tasks; 6 7 namespace Wolfy.SerializerDemo 8 { 9 public class Person 10 { 11 public Person() { } 12 public Person(string name, int age, DateTime bithday) 13 { 14 this.name = name; 15 this.age = age; 16 this.birthday = bithday; 17 } 18 private string name; 19 20 public string Name 21 { 22 get { return name; } 23 set { name = value; } 24 } 25 private int age; 26 27 public int Age 28 { 29 get { return age; } 30 set { age = value; } 31 } 32 private DateTime birthday; 33 34 public DateTime Birthday 35 { 36 get { return birthday; } 37 set { birthday = value; } 38 } 39 } 40 }
注意:Date 对象,在 JSON 中表示为“\/Date(刻度数)\/”。 刻度数是一个正的或负的长值,该值指示从 UTC 1970 年 1 月 1 日午夜开始已经过的刻度数(毫秒)。
所支持的最大日期值为 MaxValue(9999 年 12 月 31 日 11:59:59 PM),而所支持的最小日期值为 MinValue(0001 年 1 月 1 日 12:00:00 AM)。如果想得到如“2013-01-01”需要对时间进行转换:这里提供一种js转换的方式,具体方法如下:
在对Person序列化需注意:Person类要有无参的构造函数。不然在反序列化时会出现如下错误:
XmlSerializer类
将对象序列化到 XML 文档中和从 XML 文档中反序列化对象。XmlSerializer 使您得以控制如何将对象编码到 XML 中。
命名空间:System.Xml.Serialization
程序集:System.Xml(在 system.xml.dll 中)
XML 序列化是将对象的公共属性 (Property) 和字段转换为序列格式(这里是指 XML)以便存储或传输的过程。反序列化则是从 XML 输出中重新创建原始状态的对象。因此,可以将序列化视为将对象的状态保存到流或缓冲区的方法。例如,ASP.NET 使用 XmlSerializer 类对 XML Web services 消息进行编码。
对象中的数据是用编程语言构造来描述的,如类、字段、属性 (Property)、基元类型、数组,甚至 XmlElement 或 XmlAttribute 对象形式的嵌入 XML。您可以创建自己的用属性 (Attribute) 批注的类,或使用 XML 架构定义工具 (Xsd.exe) 生成基于现有 XML 架构定义 (XSD) 文档的类。如果有 XML 架构,则可以运行 XSD.exe 产生一组类,将这组类的类型强声明为此架构,并用属性 (Attribute) 批注这些类以便在序列化时遵循此架构。
在对象和 XML 之间传输数据需要从编程语言构造到 XML 架构的映射和从 XML 架构到编程语言构造的映射。XmlSerializer 和相关工具(如 Xsd.exe)在设计时和运行时都能在这两种技术之间提供一个桥梁。在设计时,使用 Xsd.exe 可从自定义类产生 XML 架构文档 (.xsd) 或从给定架构产生类。不论何种情况,这些类都用自定义属性 (Attribute) 批注,以指示 XmlSerializer 如何在 XML 架构系统和公共语言运行库之间映射。在运行时,可以将这些类的实例序列化到遵循给定架构的 XML 文档中。同样,可以将这些 XML 文档反序列化到运行时对象中。注意,XML 架构是可选的,在设计时或运行时不是必需的。
为控制所生成的 XML,可以向类和成员应用特殊属性 (Attribute)。例如,为指定不同的 XML 元素名称,可以将 XmlElementAttribute 应用于公共字段或属性 (Property),同时设置 ElementName 属性 (Property)。有关类似属性的完整列表,请参见 控制 XML 序列化的属性。也可以实现 IXmlSerializable 接口以控制 XML 输出。
如果所生成的 XML 必须符合万维网联合会 (www.w3.org) 文档“简单对象访问协议 (SOAP) 1.1”的第 5 部分,则必须利用 XmlTypeMapping 构造 XmlSerializer。若要进一步控制编码后的 SOAP XML,请使用 控制编码的 SOAP 序列化的属性 中所列的属性 (Attribute)。
有了 XmlSerializer,可以利用使用强类型类的优点并仍具有 XML 的灵活性。通过在强类型类中使用类型为 XmlElement、XmlAttribute 或 XmlNode 的字段或属性 (Property),可以将部分 XML 文档直接读入 XML 对象中。
如果使用扩展 XML 架构,则也可以使用 XmlAnyElementAttribute 和 XmlAnyAttributeAttribute 属性 (Attribute) 来序列化及反序列化在原始架构中找不到的元素或属性 (Attribute)。若要使用这些对象,请将 XmlAnyElementAttribute 应用到返回 XmlElement 对象数组的字段中,或者将 XmlAnyAttributeAttribute 应用到返回 XmlAttribute 对象数组的字段中。
如果属性 (Property) 或字段返回一个复杂对象(如数组或类实例),则 XmlSerializer 将其转换为嵌套在主 XML 文档内的元素。例如,以下代码中的第一个类返回第二个类的实例。
XmlSerializer默认的序列化规则
对于一般的数据对象(这里就用上面的Person类来说明),使用XmlSerializer对其进行序列化后会生成怎样的XML结构。首先定义一个辅助方法专门进行基于XmlSerializer的序列化操作。
1 static void Serialize<T>(T instance, string fileName) 2 { 3 using (XmlWriter writer = new XmlTextWriter(fileName, Encoding.UTF8)) 4 { 5 XmlSerializer serializer = new XmlSerializer(typeof(T)); 6 serializer.Serialize(writer, instance); 7 } 8 }
在控制台程序中,通过调用上面的Serialize<T>方法将创建的Person对象序列化保存在xml文件中。
1 static void Main(string[] args) 2 { 3 Person p = new Person("Wolfy", 24, Convert.ToDateTime("1989-04-11")); 4 Serialize<Person>(p, "person.xml"); 5 Console.Read(); 6 }
上面程序执行后,打开生成的person.xml文件,如下所示xml代表序列化后生成的xml结构。
<?xml version="1.0" encoding="utf-8"?><Person xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema"><Name>Wolfy</Name><Age>24</Age><Birthday>1989-04-11T00:00:00</Birthday></Person>
从而得出XmlSerializer在默认的情况下的序列化规则:
注意:这里同样得定义一个空的无参构造函数。这个函数是必须的,因为反序列化的时候需要调用它创建对象。如果将其从成员列表中移除,在序列化的时候会抛出InvalidOptionException异常,并提示"Person无法序列化因为它没有无参的构造函数"。
对于被序列化对象的属性,不仅要求是共有的,还要求是可读写的,如果将Person类中age和birthday分别改成只读/写的属性。如下:
1 public int Age 2 { 3 get { return age; } 4 //set { age = value; } 5 } 6 private DateTime birthday; 7 8 public DateTime Birthday 9 { 10 //get { return birthday; } 11 set { birthday = value; } 12 }
执行控制台程序,执行结果如下,可见只读、写的age和birthday并没有出现在序列化的xml中。
<?xml version="1.0" encoding="utf-8"?><Person xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema"><Name>Wolfy</Name></Person>
如果是自定义自动实现的属性,定义成下面的这两种方式,Person对象是不能被序列化的
通过定制XmlSerializer控制XML结构
上面列出的仅仅是XmlSerializer采用的默认序列化规则。在很多情况下,我们需要序列化生成的XML符合一个既定的XSD,那么就需要对xmlSerializer的序列化进行人工的干预和控制。
对最终生成的XML结构的控制可以通过在数据类型和它的字段/属性成员上应用相应的特性来实现。
我们可以按照如下的方式在类型上应用XmlRootAttribute特性对ElementName和Namespace属性进行相应的设置最终控制根节点的元素名称和命名空间。在属性Name上应用XmlAttributeAttrubute特性让该属性以XML属性的形式被序列化。
生成的xml文件:
反序列化通用方法:
1 static T Desirialize<T>(string fileName) 2 { 3 using (FileStream fs = new FileStream(fileName, FileMode.Open)) 4 { 5 XmlSerializer serializer = new XmlSerializer(typeof(T)); 6 return (T)serializer.Deserialize(fs); 7 } 8 9 }
参考文献:
MSDN:
http://msdn.microsoft.com/zh-cn/library/system.xml.serialization.xmlserializer(VS.80).aspx
http://msdn.microsoft.com/en-us/library/system.web.script.serialization.javascriptserializer.aspx
《WCF全面解析 上册》蒋金楠 著 电子工业出版社
关于序列化先总结到这里,关于DataContractSerializer之后会用专门一篇来总结。
自己不是大牛,但是也说的上一个进击的小菜......
软件产品常常会出现这样的情况:产品性能因某些无法预料的瓶颈而受到干扰,导致程序的处理效率降低,性能得不到充分的发挥。如何快速有效地找到软件产品的性能瓶颈,则是我们感兴趣的内容之一。
在本文中,我将解释我如何清理和替换重复、 混乱遍布许多方法在应用程序中的代码使用ASP.NET Web API 的筛选器来完成ASP.NET Web API 接口执行时间监控。我们的项目中有如下的需求:我的工作相关的项目 (使用 ASP.NET Web API 框架) 要求记录下服务接口的调用执行时间以及请求延迟、 故障率每秒的请求总数,以帮助运营团队。
Web API 筛选器是你放到Action上的自定义属性方法 (或Controller) 添加常用功能。 Web API 筛选使您得以添加预处理和后处理的行为,本文的代码来自于How to intercept all the ASP.NET WebApi controller action methods calls with Ninject interception for logging? 和 Log duration of an ASP Web API action 。
下面是详细代码:
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Web;
using System.Web.Http.Controllers;
using System.Web.Http.Filters;
namespace ContactManager.Filters
{
public class TimingActionFilter : ActionFilterAttribute
{
private const string Key = "__action_duration__";
public override void OnActionExecuting(HttpActionContext actionContext)
{
if (SkipLogging(actionContext))
{
return;
}
var stopWatch = new Stopwatch();
actionContext.Request.Properties[Key] = stopWatch;
stopWatch.Start();
}
public override void OnActionExecuted(HttpActionExecutedContext actionExecutedContext)
{
if (!actionExecutedContext.Request.Properties.ContainsKey(Key))
{
return;
}
var stopWatch = actionExecutedContext.Request.Properties[Key] as Stopwatch;
if (stopWatch != null)
{
stopWatch.Stop();
var actionName = actionExecutedContext.ActionContext.ActionDescriptor.ActionName;
var controllerName = actionExecutedContext.ActionContext.ActionDescriptor.ControllerDescriptor.ControllerName;
Debug.Print(string.Format("[Execution of{0}- {1} took {2}.]", controllerName, actionName, stopWatch.Elapsed));
}
}
private static bool SkipLogging(HttpActionContext actionContext)
{
return actionContext.ActionDescriptor.GetCustomAttributes<NoLogAttribute>().Any() ||
actionContext.ControllerContext.ControllerDescriptor.GetCustomAttributes<NoLogAttribute>().Any();
}
}
[AttributeUsage(AttributeTargets.Method | AttributeTargets.Class, Inherited = true)]
public class NoLogAttribute : Attribute
{
}
}
然后在代码里注册 GlobalConfiguration.Configuration.Filters.Add(new TimingActionFilter());