WCF DataContract标记去除方式解析

开发 开发工具
WCF DataContract标记的应用在实际编程开发中有时会比较麻烦,那么如何才能省略掉这一标记又不影响程序开发呢?在这里将会一一介绍。

WCF是一个基于.Net Framework 2.0基础的由微软公司研发的开发插件,它的实际应用可以帮助编程人员轻松实现特定的功能需求。在这里我们将会介绍一种应用技巧,就是如何在使用中省略掉WCF DataContract标记。#t#

写过WCF程序的朋友都知道,在对实体对象在WCF和客户端之间传递时一定要加WCF DataContract标记这个类并用DataMember来标记要序列化的属性/字段。这一直正确,只是在.NET Framework 3.5 SP1中新添加了一些支持,那就是你不一定必须对这些实体对象应用DataContract标记,这被称作对plain old C# objects(POCO)的序列化支持。

 

Serializable标记大家都很熟悉,它是XmlSerializer的标记,在WCF中其实很少用这个标记,因为我们WCF用的是DataContractSerializer,对应的标记也是DataContract。但对于SP1来说,Serializable也以XmlSerializer的规则被正常解析,其对应的Mapping规则和Serializer对应,其公有可读写字段被默认序列化。当然,你也可以通过XmlElement等标记来做高级映射,但这不是我们这里需要谈及的内容。

 

WCF DataContract标记对应的序列化处理叫做DataContactSerializer。在WCF中一旦一个类被标记为DataContract,那么只有标记为DataMember的字段/属性才会被序列化。但如果你使用DataContract标记,那么DataContractSerializer默认将所有公有可读写字段序列化(这和Serializable是一样的)。假设我们有这么一个类:

 

  1. public class Person  
  2. {  
  3. public Person()  
  4. { }  
  5. public Person(string strId, string strName)  
  6. {  
  7. this.Id = strId;  
  8. this.Name = strName;  
  9. }  
  10. private string strid;  
  11. public string Id { get { return strid; } set { strid = value; } }  
  12. public string Name;  
  13. public Person Spouse;  
  14. private int Number = 343;  

 

 

对于DataSerializer来说,他和给所有公有属性添加DataMember并将类标记为WCF DataContract标记是一样的。下面的一段程序分别将一个Person的实例对象分别用XmlSerializer和DataContractSerializer来序列化:

 

  1. static void Main(string[] args)  
  2. {  
  3. Person p = new Person();  
  4. p.Id = "123";  
  5. p.Name = "Aaron";  
  6. p.Spouse = new Person();  
  7. p.Spouse.Id = "456";  
  8. p.Spouse.Name = "Monica";  
  9. DataContractSerializer dcs = new 
    DataContractSerializer(typeof(Person));  
  10. using (FileStream fs = new FileStream("person.xml", 
    FileMode.Create))  
  11. {  
  12. dcs.WriteObject(fs, p);  
  13. }  
  14. XmlSerializer xs = new XmlSerializer(typeof(Person));  
  15. using (FileStream fs = new FileStream("person_serialization.xml",
     FileMode.Create))  
  16. {  
  17. xs.Serialize(fs, p);  
  18. }  

 

对于序列化后的内容我们得到的结果其实是一样的:仅有公有属性/字段被序列化

 

  1. < Person xmlns="http://schemas.datacontract.org/2004/07/
    Serialization"
     xmlns:i="http://www.w3.org/2001/XMLSchema-instance"> 
  2. < Id>123< /Id> 
  3. < Name>Aaron< /Name> 
  4. < Spouse> 
  5. < Id>456< /Id> 
  6. < Name>Monica< /Name> 
  7. < Spouse i:nil="true"/> 
  8. < /Spouse> 
  9. < /Person> 

 

但如果你使用了DataContract来标记这个类,却没有使用DataMember,那么没有任何属性/字段被序列化:

 

  1. [DataContract]  
  2. public class Person  
  3. {  
  4. private string strid;  
  5. public string Id { get { return strid; } set { strid = value; } }  
  6. public string Name;  
  7. public Person Spouse;  
  8. private int Number = 343;  
  9. }  
  10. < Person xmlns="http://schemas.datacontract.org/2004/07/
    Serialization"
     xmlns:i=http://www.w3.org/2001/XMLSchema-instance /> 

 

 

对于将类标记成Serializable,DataContractSerializer的序列化可能让我们觉得有些奇怪,它本质上是将所有可读写字段序列化,这其中还包括私有字段。例如我们将Person类用[Serializable]标记,执行程序,我们会得到以下的结果:

 

  1. < Person xmlns="http://schemas.datacontract.org/2004/07
    /Serialization"
     xmlns:i="http://www.w3.org/2001/XMLSchema-instance"> 
  2. < Name>Aaron< /Name> 
  3. < Number>0< /Number> 
  4. < Spouse> 
  5. < Name>Monica< /Name> 
  6. < Number>0< /Number> 
  7. < Spouse i:nil="true"/> 
  8. < _id>456< /_id> 
  9. < /Spouse> 
  10. < _id>123< /_id> 
  11. < /Person> 

 

一个简单的WCF程序来看看来检验一下是否正确。在Contract生命中我们并不需要制定任何的标记,并声明一个得到DeskMesh的方法:

 

  1. [ServiceContract]  
  2. public interface IDeskMesh  
  3. {  
  4. [OperationContract]  
  5. DeskMesh GetDeskMesh(string name);  
  6. }  
  7. public class DeskMesh  
  8. {  
  9. private int _id;  
  10. private int Number = 4433;  
  11. public int ID  
  12. {  
  13. get { return _id; }  
  14. set { _id = value; }  
  15. }  
  16. public string Name { get; set; }  
  17. public string Description { get; set; }  
  18. public string Unit { get; set; }  
  19. public float Price { get; set; }  
  20. public DateTime Created { get; set; }  
  21. public override string ToString()  
  22. {  
  23. return string.Format("ID:{4}"r"nName: {0}"r"nUnit:{1}"r"
    nPrice:{2}"r"nCreated:{3}"r"nNumber:{5}",  
  24. Name, Unit, Price, Created.ToShortDateString(),
    ID.ToString(),Number.ToString());  
  25. }  

 

 

客户端调用,会返回一个DeskMesh的实例。通过结果,你会发现这完全和你标记WCF DataContract标记的实体在WCF两端传递一模一样。

 

  1. void Main(string[] args)  
  2. {  
  3. Console.WriteLine("Requesting...");  
  4. ServiceClient client = new ServiceClient();  
  5. DeskMesh mesh = client.GetDeskMesh("");  
  6. Console.WriteLine(mesh.ToString());  
  7. Console.WriteLine("press any key to continue...");  
  8. Console.Read();  

 

 

总结一下吧,WCF中应用各个标记时所作的序列化处理:

1. 不给任何标记将会做XML映射,所有公有属性/字段都会被序列化

2. [Serializable]标记会将所有可读写字段序列化

3. [DataContract]和[DataMember]联合使用来标记被序列化的字段

以上就是对WCF DataContract标记的相关介绍。

责任编辑:曹凯 来源: 博客园
相关推荐

2010-03-02 16:09:53

WCF框架

2009-12-22 13:25:58

WCF性能计数器内存

2010-01-20 17:41:13

VB.NET标记语句

2010-02-26 14:05:57

WCF通信方式

2009-12-22 15:14:33

WCF调用

2010-02-22 14:18:34

WCF服务验证

2010-02-23 10:51:32

WCF Address

2009-11-05 16:12:09

WCF会话服务

2010-02-22 14:53:17

WCF用户密码

2009-12-07 09:58:35

WCF性能

2009-12-07 09:32:44

WCF开源

2010-02-22 10:18:18

WCF配置文件

2009-11-30 16:24:24

PHP脚本

2010-02-24 12:41:58

WCF异常处理

2010-03-02 09:32:54

WCF服务消息

2010-02-24 10:07:48

WCF跨越边界

2010-02-26 17:44:51

WCF安全参数

2010-02-22 17:21:02

WCF消息交换

2010-03-02 16:05:48

WCF端点配置

2010-02-26 14:12:27

WCF元数据
点赞
收藏

51CTO技术栈公众号