当前位置: 首页 > news >正文

艾伟:[原创]谈谈WCF中的Data Contract(2):WCF Data Contract对Generic的支持

通过第一部分的介绍,我们可以体会到,WCF Data ContractCLR TypeNeutral Contract之间搭建了一座桥梁,弥合了.NET世界和厂商中立世界的差异。通过WCF Data Contract我们将CLR Data Type暴露成一个厂商中立的数据结构的描述,同样通过WCF Data Contract我们将一个现有的CLR Data Type和既定的Neutral contract进行适配。

.NET中,基于Primary Type,比如Int32String等等,他们具有一个简单的默认的序列化方式和结构,可以说他们不需要Data Contract。接下来我们主要讨论的是一些相对比较特殊的、完全基于.NETData Type,比如GenericCollection,和Dictionary。首先,我们结合例子来谈谈基于GenericData TypeData Contract

假设我们需要创建一个用于处理一些单据(Bill)的Service,比如如Order BillSales Bill等。一般的单据都有一个单据头(Header)和明细(Detail)列表,为此我们创建了一个GenericBill。并

namespace  Artech.SpecialDataContract.Contract
{
    [DataContract]
    
public class Bill<THeader, TDetail>
    
{
        [DataMember]
        
public THeader Header
        
getset; }

        [DataMember]
        
public IList<TDetail> DetailList
        
getset; }
    }


    [DataContract]
    
public class OrderHeader
    
{
        [DataMember]
        
public Guid OrderID
        
getset; }

        [DataMember]
        
public DateTime OrderDate
        
getset; }
    }


    [DataContract]
    
public class OrderDetail
    
{
        [DataMember]
        
public Guid ProductID
        
getset; }

        [DataMember]
        
public int Quantity
        
getset; }
    }


}

为处理订单单据创建了机遇订单的HeaderDetail

对于一个
Neutral Service ContractNeutral Data Contract本身是不可能支持Generic的,也就是Neutral Contract只能是对一个具体的CLR Type的体现。所以在定义Service Contract的时候,对于那些包含Generic Type作为参数或者返回值得Operation,我们必须指定一个具体的Data Type。所以我们创建了如下一个IBillManager Service Contract

 

 

namespace  Artech.SpecialDataContract.Contract
{
    [ServiceContract]
    
public interface IBillManager
    
{
        [OperationContract]
        
void Procss(Bill<OrderHeader, OrderDetail> orderBill);
    }

}

如何我们现在

Host 基于这样一个 Contract Service ,你猜我们作为参数的数据类型将会如何体现的。

通过WSDL,我们会发现该ServiceData Contract将会以下面一段XSD的方式来呈现:

 

<? xml version="1.0" encoding="utf-8"  ?>
< xs:schema  elementFormDefault ="qualified"  targetNamespace ="http://schemas.datacontract.org/2004/07/Artech.SpecialDataContract.Contract"
    xmlns:xs
="http://www.w3.org/2001/XMLSchema"  xmlns:tns ="http://schemas.datacontract.org/2004/07/Artech.SpecialDataContract.Contract"
    xmlns:ser
="http://schemas.microsoft.com/2003/10/Serialization/" >
  
< xs:import  schemaLocation ="http://artech/Artech.SpecialDataContract/BillManagerService.svc?xsd=xsd1"
      namespace
="http://schemas.microsoft.com/2003/10/Serialization/"   />
  
< xs:complexType  name ="BillOfOrderHeaderOrderDetailLZ9Dq20o" >
    
< xs:annotation >
      
< xs:appinfo >
        
< GenericType  Name ="BillOf{0}{1}{#}"  Namespace ="http://schemas.datacontract.org/2004/07/Artech.SpecialDataContract.Contract"
            xmlns
="http://schemas.microsoft.com/2003/10/Serialization/" >
          
< GenericParameter  Name ="OrderHeader"  Namespace ="http://schemas.datacontract.org/2004/07/Artech.SpecialDataContract.Contract" />
          
< GenericParameter  Name ="OrderDetail"  Namespace ="http://schemas.datacontract.org/2004/07/Artech.SpecialDataContract.Contract" />
        
</ GenericType >
      
</ xs:appinfo >
    
</ xs:annotation >
    
< xs:sequence >
      
< xs:element  minOccurs ="0"  name ="DetailList"  nillable ="true"  type ="tns:ArrayOfOrderDetail" />
      
< xs:element  minOccurs ="0"  name ="Header"  nillable ="true"  type ="tns:OrderHeader" />
    
</ xs:sequence >
  
</ xs:complexType >
  
< xs:element  name ="BillOfOrderHeaderOrderDetailLZ9Dq20o"  nillable ="true"  type ="tns:BillOfOrderHeaderOrderDetailLZ9Dq20o" />
  
< xs:complexType  name ="ArrayOfOrderDetail" >
    
< xs:sequence >
      
< xs:element  minOccurs ="0"  maxOccurs ="unbounded"  name ="OrderDetail"  nillable ="true"  type ="tns:OrderDetail" />
    
</ xs:sequence >
  
</ xs:complexType >
  
< xs:element  name ="ArrayOfOrderDetail"  nillable ="true"  type ="tns:ArrayOfOrderDetail" />
  
< xs:complexType  name ="OrderDetail" >
    
< xs:sequence >
      
< xs:element  minOccurs ="0"  name ="ProductID"  type ="ser:guid" />
      
< xs:element  minOccurs ="0"  name ="Quantity"  type ="xs:int" />
    
</ xs:sequence >
  
</ xs:complexType >
  
< xs:element  name ="OrderDetail"  nillable ="true"  type ="tns:OrderDetail" />
  
< xs:complexType  name ="OrderHeader" >
    
< xs:sequence >
      
< xs:element  minOccurs ="0"  name ="OrderDate"  type ="xs:dateTime" />
      
< xs:element  minOccurs ="0"  name ="OrderID"  type ="ser:guid" />
    
</ xs:sequence >
  
</ xs:complexType >
  
< xs:element  name ="OrderHeader"  nillable ="true"  type ="tns:OrderHeader" />
</ xs:schema >

对于不习惯看XSD的朋友,我们可以通过Add Service Reference的方式创建本地的Proxy file,借助生成的与之对应的Class来看看这个XSD最终呈现的结构:

    [System.Diagnostics.DebuggerStepThroughAttribute()]
    [System.CodeDom.Compiler.GeneratedCodeAttribute(
" System.Runtime.Serialization " " 3.0.0.0 " )]
    [System.Runtime.Serialization.DataContractAttribute(Name
= " BillOfOrderHeaderOrderDetailLZ9Dq20o " , Namespace = " http://schemas.datacontract.org/2004/07/Artech.SpecialDataContract.Contract " )]
    [System.SerializableAttribute()]
    
public  partial  class  BillOfOrderHeaderOrderDetailLZ9Dq20o :  object , System.Runtime.Serialization.IExtensibleDataObject, System.ComponentModel.INotifyPropertyChanged  {
        
        [System.NonSerializedAttribute()]
        
private System.Runtime.Serialization.ExtensionDataObject extensionDataField;
        
        [System.Runtime.Serialization.OptionalFieldAttribute()]
        
private Artech.SpecialDataContract.Client.BillManagerService.OrderDetail[] DetailListField;
        
        [System.Runtime.Serialization.OptionalFieldAttribute()]
        
private Artech.SpecialDataContract.Client.BillManagerService.OrderHeader HeaderField;
        
        [global::System.ComponentModel.BrowsableAttribute(
false)]
        
public System.Runtime.Serialization.ExtensionDataObject ExtensionData {
            
get {
                
return this.extensionDataField;
            }

            
set {
                
this.extensionDataField = value;
            }

        }

        
        [System.Runtime.Serialization.DataMemberAttribute()]
        
public Artech.SpecialDataContract.Client.BillManagerService.OrderDetail[] DetailList {
            
get {
                
return this.DetailListField;
            }

            
set {
                
if ((object.ReferenceEquals(this.DetailListField, value) != true)) {
                    
this.DetailListField = value;
                    
this.RaisePropertyChanged("DetailList");
                }

            }

        }

        
        [System.Runtime.Serialization.DataMemberAttribute()]
        
public Artech.SpecialDataContract.Client.BillManagerService.OrderHeader Header {
            
get {
                
return this.HeaderField;
            }

            
set {
                
if ((object.ReferenceEquals(this.HeaderField, value) != true)) {
                    
this.HeaderField = value;
                    
this.RaisePropertyChanged("Header");
                }

            }

        }

        
        
public event System.ComponentModel.PropertyChangedEventHandler PropertyChanged;
        
        
protected void RaisePropertyChanged(string propertyName) {
            System.ComponentModel.PropertyChangedEventHandler propertyChanged 
= this.PropertyChanged;
            
if ((propertyChanged != null)) {
                propertyChanged(
thisnew System.ComponentModel.PropertyChangedEventArgs(propertyName));
            }

        }

    }

    
    [System.Diagnostics.DebuggerStepThroughAttribute()]
    [System.CodeDom.Compiler.GeneratedCodeAttribute(
" System.Runtime.Serialization " " 3.0.0.0 " )]
    [System.Runtime.Serialization.DataContractAttribute(Name
= " OrderHeader " , Namespace = " http://schemas.datacontract.org/2004/07/Artech.SpecialDataContract.Contract " )]
    [System.SerializableAttribute()]
    
public  partial  class  OrderHeader :  object , System.Runtime.Serialization.IExtensibleDataObject, System.ComponentModel.INotifyPropertyChanged  {
        
        [System.NonSerializedAttribute()]
        
private System.Runtime.Serialization.ExtensionDataObject extensionDataField;
        
        [System.Runtime.Serialization.OptionalFieldAttribute()]
        
private System.DateTime OrderDateField;
        
        [System.Runtime.Serialization.OptionalFieldAttribute()]
        
private System.Guid OrderIDField;
        
        [global::System.ComponentModel.BrowsableAttribute(
false)]
        
public System.Runtime.Serialization.ExtensionDataObject ExtensionData {
            
get {
                
return this.extensionDataField;
            }

            
set {
                
this.extensionDataField = value;
            }

        }

        
        [System.Runtime.Serialization.DataMemberAttribute()]
        
public System.DateTime OrderDate {
            
get {
                
return this.OrderDateField;
            }

            
set {
                
if ((this.OrderDateField.Equals(value) != true)) {
                    
this.OrderDateField = value;
                    
this.RaisePropertyChanged("OrderDate");
                }

            }

        }

        
        [System.Runtime.Serialization.DataMemberAttribute()]
        
public System.Guid OrderID {
            
get {
                
return this.OrderIDField;
            }

            
set {
                
if ((this.OrderIDField.Equals(value) != true)) {
                    
this.OrderIDField = value;
                    
this.RaisePropertyChanged("OrderID");
                }

            }

        }

        
        
public event System.ComponentModel.PropertyChangedEventHandler PropertyChanged;
        
        
protected void RaisePropertyChanged(string propertyName) {
            System.ComponentModel.PropertyChangedEventHandler propertyChanged 
= this.PropertyChanged;
            
if ((propertyChanged != null)) {
                propertyChanged(
thisnew System.ComponentModel.PropertyChangedEventArgs(propertyName));
            }

        }

    }

    
    [System.Diagnostics.DebuggerStepThroughAttribute()]
    [System.CodeDom.Compiler.GeneratedCodeAttribute(
" System.Runtime.Serialization " " 3.0.0.0 " )]
    [System.Runtime.Serialization.DataContractAttribute(Name
= " OrderDetail " , Namespace = " http://schemas.datacontract.org/2004/07/Artech.SpecialDataContract.Contract " )]
    [System.SerializableAttribute()]
    
public  partial  class  OrderDetail :  object , System.Runtime.Serialization.IExtensibleDataObject, System.ComponentModel.INotifyPropertyChanged  {
        
        [System.NonSerializedAttribute()]
        
private System.Runtime.Serialization.ExtensionDataObject extensionDataField;
        
        [System.Runtime.Serialization.OptionalFieldAttribute()]
        
private System.Guid ProductIDField;
        
        [System.Runtime.Serialization.OptionalFieldAttribute()]
        
private int QuantityField;
        
        [global::System.ComponentModel.BrowsableAttribute(
false)]
        
public System.Runtime.Serialization.ExtensionDataObject ExtensionData {
            
get {
                
return this.extensionDataField;
            }

            
set {
                
this.extensionDataField = value;
            }

        }

        
        [System.Runtime.Serialization.DataMemberAttribute()]
        
public System.Guid ProductID {
            
get {
                
return this.ProductIDField;
            }

            
set {
                
if ((this.ProductIDField.Equals(value) != true)) {
                    
this.ProductIDField = value;
                    
this.RaisePropertyChanged("ProductID");
                }

            }

        }

        
        [System.Runtime.Serialization.DataMemberAttribute()]
        
public int Quantity {
            
get {
                
return this.QuantityField;
            }

            
set {
                
if ((this.QuantityField.Equals(value) != true)) {
                    
this.QuantityField = value;
                    
this.RaisePropertyChanged("Quantity");
                }

            }

        }

        
        
public event System.ComponentModel.PropertyChangedEventHandler PropertyChanged;
        
        
protected void RaisePropertyChanged(string propertyName) {
            System.ComponentModel.PropertyChangedEventHandler propertyChanged 
= this.PropertyChanged;
            
if ((propertyChanged != null)) {
                propertyChanged(
thisnew System.ComponentModel.PropertyChangedEventArgs(propertyName));
            }

        }

    }

为了使大家一眼就能了解整个结构,我对上面的代码进行简化:

 

namespace  Artech.SpecialDataContract.Client.BillManagerService
{
    [DataContract]
    
public class BillOfOrderHeaderOrderDetailLZ9Dq20o
    
{
        [DataMember]
        
public OrderHeader Header 
        
getset; }

        [DataMember]
        
public OrderDetail[] DetailList
        
getset; }
    }


    [DataContract]
    
public class OrderHeader
    
{
        [DataMember]
        
public Guid OrderID
        
getset; }

        [DataMember]
        
public DateTime OrderDate
        
getset; }
    }


    [DataContract]
    
public class OrderDetail
    
{
        [DataMember]
        
public Guid ProductID
        
getset; }

        [DataMember]
        
public int Quantity
        
getset; }
    }


}

我们可以通过上面的code,注意到下面的细节:

·         Generic class Bill<THeader, TDetail>没有了,取而代之的是使用了具体OrderHeaderOrderDetial的新的非Generic classBillOfOrderHeaderOrderDetailLZ9Dq20o。正如我们在上面所说,Neutral Contract根本就不知道Generic为何物。

·         新的Class name的名称很难看,它有下面几个部分组成:BillGeneric Type Name+ Of + OrderHeaderGeneric Type的第一个类型参数对应的具体类型名称)+OrderDetailGeneric Type的第二个类型参数对应的具体类型名称)+lLZ9Dq20oGeneric Type参数类型NamespaceHash Value)。

·         原本使用IList表示的DetailList变成了Arraypublic OrderDetail[] DetailList),这个将在和面的部分介绍。

我想你也不能容忍生成的如此冗长、甚至没有太大意义的Class name。我们有办法生成一个友好的名称。那就是显示指定Data ContractName

[DataContract(Name = " OrderBill " )]
    
public   class  Bill < THeader, TDetail >
    
{
        [DataMember]
        
public THeader Header
        
getset; }

        [DataMember]
        
public IList<TDetail> DetailList
        
getset; }
}

现在对应的Data Contract Name将变成我们指定的名称。

public partial class OrderBill : object, System.Runtime.Serialization.IExtensibleDataObject, System.ComponentModel.INotifyPropertyChanged {}

但是上面的做法是不对的。原因很简单,如果我们在Service中添加一个处理Sales BillOperation(当然我们会添加两个额外的HeaderDetailSalesHeaderSalesDetail):

    [ServiceContract]
    
public   interface  IBillManager
    
{
        [OperationContract(Name
=”ProcessOrderBill”)]
        
void Procss(Bill<OrderHeader, OrderDetail> orderBill);

        [OperationContract(Name 
=”ProcessSalesBill”)]
        
void Procss(Bill<SalesHeader, SalesDetail> salesBill);
    }

很显然,WCF需要为Order BillSales Bill创建两个Data Contract,但是现在你却把他们的名称显示地限定到一个固定的名称,很显然这会造成命名的冲突。如果你通过Browser试图访问Service,你会得到如下的Error

The server was unable to process the request due to an internal error. For more information about the error, either turn on IncludeExceptionDetailInFaults (either from ServiceBehaviorAttribute or from the <serviceDebug> configuration behavior) on the server in order to send the exception information back to the client, or turn on tracing as per the Microsoft .NET Framework 3.0 SDK documentation and inspect the server trace logs.

所以WCF需要为此提供这样的机制:基于不同的泛型类型参数生成不同Data Contract Name,这样才能解决命名冲突。我们可以稍微修改一下Data Contract 的定义就可以了:

    [DataContract(Name = " Bill_{0}_{1} " )]
    
public   class  Bill < THeader, TDetail >
    
{
        [DataMember]
        
public THeader Header
        
getset; }

        [DataMember]
        
public IList<TDetail> DetailList
        
getset; }
}

其中{0}{1}分别代表第一个泛型类型参数和第二个泛型类型参数的名称,一次类推,你可以根据参数类型的个数设置{2}{3}…

这样我们生成的两个DataContract的名称为:Bill_OrderHeader_OrderDetailBill_SalesHeader_SalesDetail

但是这并没有根本解决问题,如果在我现在不同的Namespace中创建了两个OrderHeaderOrderDetail呢?这无疑在.NET中是合法的,但是对于DataContract有有可能出现命名冲突。

这也就是为什么WCF默认机制下会为Data Contract Name添加一个额外hash value的原因。其实你也可以以你自己的方式添加这个Hash value

 

 

[DataContract(Name = " Bill_{0}_{1}_{#} " )]
    
public   class  Bill < THeader, TDetail >
    
{
        [DataMember]
        
public THeader Header
        
getset; }

        [DataMember]
        
public IList<TDetail> DetailList
        
getset; }
}

上面的

{#} 就代表这样一个 Hash Value, 我想你会想到现在生成的 Data Contract Name 象什么样子: Bill_OrderHeader_OrderDetail_LZ9Dq20o Bill_SalesHeader_SalesDetail_LZ9Dq20o

相关文章:

  • NeHe OpenGL第三十四课:地形
  • ansible-playbook批量部署zabbix
  • 艾伟_转载:C#来创建和读取XML文档
  • 一例HP ADG数据恢复成功(8×73GB SCSI)
  • 艾伟_转载:LINQ to SQL、NHibernate比较(二)-- LINQ to SQL实例
  • php经典面试题
  • JavaScript控制图片横向滚动代码
  • 关于服务器缓存的思考
  • 伸向开源世界的“橄榄枝”
  • 运营中心:网站分析、网络广告注释
  • Fedora8上Apache Httpd与Tomcat6初集成
  • 转贴:下辈子,无论爱与不爱,都不会再见
  • 塔式,机架,刀片服务器
  • 静态网页和动态网页
  • 安全运维之:服务器遭受攻击后的一般处理过程
  • Angular6错误 Service: No provider for Renderer2
  • Git 使用集
  • go语言学习初探(一)
  • Java多态
  • Median of Two Sorted Arrays
  • MYSQL如何对数据进行自动化升级--以如果某数据表存在并且某字段不存在时则执行更新操作为例...
  • Python利用正则抓取网页内容保存到本地
  • Spring Cloud(3) - 服务治理: Spring Cloud Eureka
  • 二维平面内的碰撞检测【一】
  • 关于 Linux 进程的 UID、EUID、GID 和 EGID
  • 离散点最小(凸)包围边界查找
  • 三栏布局总结
  • 数组的操作
  • 赢得Docker挑战最佳实践
  • Unity3D - 异步加载游戏场景与异步加载游戏资源进度条 ...
  • 阿里云重庆大学大数据训练营落地分享
  • 带你开发类似Pokemon Go的AR游戏
  • ​LeetCode解法汇总518. 零钱兑换 II
  • ​Python 3 新特性:类型注解
  • ​用户画像从0到100的构建思路
  • #NOIP 2014# day.1 生活大爆炸版 石头剪刀布
  • #我与Java虚拟机的故事#连载16:打开Java世界大门的钥匙
  • (1)安装hadoop之虚拟机准备(配置IP与主机名)
  • (12)Hive调优——count distinct去重优化
  • (2)STM32单片机上位机
  • (C语言)fgets与fputs函数详解
  • (Matalb时序预测)PSO-BP粒子群算法优化BP神经网络的多维时序回归预测
  • (大众金融)SQL server面试题(1)-总销售量最少的3个型号的车及其总销售量
  • (分布式缓存)Redis哨兵
  • (附源码)spring boot车辆管理系统 毕业设计 031034
  • (九)One-Wire总线-DS18B20
  • (七)MySQL是如何将LRU链表的使用性能优化到极致的?
  • (原創) 博客園正式支援VHDL語法著色功能 (SOC) (VHDL)
  • (转载)跟我一起学习VIM - The Life Changing Editor
  • (转载)虚幻引擎3--【UnrealScript教程】章节一:20.location和rotation
  • .net core 实现redis分片_基于 Redis 的分布式任务调度框架 earth-frost
  • .NET Core实战项目之CMS 第一章 入门篇-开篇及总体规划
  • .net 简单实现MD5
  • .net连接MySQL的方法
  • /etc/sudoers (root权限管理)