WCF中的集合类型

在.net中,各种类型的集合均实现了IEnumerable或者IEnumerable<T>接口,一个数据契约的数据成员可以是一个集合类型,服务契约也可以定义直接与集合交互的操作,但是这都是.net所特有的,WCf不能在元数据中公开它们,然后WCF专门为集合提供了编组原则。

     在定义服务时候,无论使用的是IEnumerable<T>、IList<T>和ICollection<T>,他们的传输形式都使用了数组,如

namespace ContractInterface

{

    [ServiceContract]

public interface IContractManager

    {

        [ OperationContract]

IEnumerable<Contract> GetContracts();

    }

}

导出的结果为

public interface IContractManager

{

Contract[] GetContracts();

    }

如果契约中的集合为具体集合类型而非接口,而且属于可序列化集合,那么只要提供的集合包含了Add方法,并且符合以下签名中的一种,WCF就能够自动地将集合规范为数组类型。

     并非只有内建的集合类型才具有自动编组为数组的能力,任何自定义的集合只要符合相同的先决条件,都可以被编组为数组。

CollectionDataContract特性

前面所示的编组为具体类型的机制并不理想,原有有三

  1. 它要求集合必须可序列化,而不是使用DataContract特性。
  2. 当服务的一方处理集合类型时,服务的另外一方若处理数组类型,就会导致双方语义不对称,集合拥有数组不具备的优势。
  3. 对于集合是否包含Add方法,或者集合是否支持IEnumerable和IEnumerable<T>接口,并没有编译时或者运行时的有效验证。如果不符合条件,就会导致数据契约不可工作。

基于以上几点,WCF提供了另外一种有效的解决方案,一种专用的特性:CollectionDataContractAttribute,定义为:

    [AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct, Inherited = false , AllowMultiple = false)]

public sealed class CollectionDataContractAttribute : Attribute

    {

public CollectionDataContractAttribute();

public bool IsReference { get; set; }

public string ItemName { get; set; }

public string KeyName { get; set; }

public string Name { get; set; }

public string Namespace { get; set; }

public string ValueName { get; set; }

    }

CollectionDataContract和DataContract相似,它不能序列化集合,将它应用到一个集合上时,会将集合当作一个泛型的链表类型公开给客户端链表可能不会对原来的集合执行任何操作,但它会提供一个类似接口的集合类型,而不是数组。例如,定义如下类型:

namespace ContractInterface

{

    [CollectionDataContract(Name= "MyCollectionOf{0}")]

public class MyCollection<T>: IEnumerable<T>

    {

public void Add(T item)

        { }

public IEnumerator<T> GetEnumerator()

        {

throw new NotImplementedException();

        }

    }

}

服务契约修改为:

namespace ContractInterface

{

    [ServiceContract]

public interface IContractManager

    {

        [ OperationContract]

List<Contract> GetContracts();

[OperationContract ]

MyCollection<Contract> GetContract();

    }

}

则,导出的客户端定义为:

MyCollectionOfContract GetContract();

集合类重新编组为

public class MyCollectionOfContract : System.Collections.Generic.List<Contract > {

    }

     在装载服务时,特性会检查Add方法和IEnumerable或者IEnumerable<T>是否存在,如果不存在而引发InvalidDataContractException异常。

     注意,不能同时将DataContract和CollecctionDataContract应用到集合类型上,在装载服务的时候同样会检查这一点。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏流柯技术学院

TestNG 三 测试方法

测试方法是可以带有参数的。每个测试方法都可以带有任意数量的参数,并且可以通过使用TestNG的@Parameters向方法传递正确的参数。

753
来自专栏区块链入门

【易错概念】Solidity语法的重载,继承的定义

以太坊智能合约语言Solitidy是一种面向对象的语言,本文结合面向对象语言的特性,讲清楚Solitidy语言的多态(Polymorphism)(重写,重载),...

1063
来自专栏Java技术栈

面试必备:30 个 Java 集合面试问题及答案

Java集合框架为Java编程语言的基础,也是Java面试中很重要的一个知识点。这里,我列出了一些关于Java集合的重要问题和答案。

862
来自专栏Java3y

Java集合总结【面试题+脑图】,将知识点一网打尽!

1815
来自专栏一枝花算不算浪漫

Java中常见数据结构Map之LinkedHashMap

2583
来自专栏java一日一条

Java 集合框架 HashSet 和 HashMap 源码剖析

之所以把HashSet和HashMap放在一起讲解,是因为二者在Java里有着相同的实现,前者仅仅是对后者做了一层包装,也就是说HashSet里面有一个Hash...

552
来自专栏IT大咖说

深入学习 Java 序列化

1824
来自专栏跟着阿笨一起玩NET

工作中遇到的问题总结1

上述用于UP_refund_Info_CaseAPIRefund_GetList是一个存储过程。

423
来自专栏阿杜的世界

Java泛型基础(一)目的泛型类总结

利用Java开发的时候,肯定会有一个类持有另一个或几个类的情况,在编写一些比较基础的组件,例如缓存操作组件,这类组件的逻辑差不多,但是希望能够处理不同的类型。

371
来自专栏10km的专栏

jface databinding: 创建readonly(只读)可观察对象(observable)

java与C++有一点不同:C++有const关键字,使用const关键字,用于指定一个参数、成员变量或函数是只读不可修改的,通过const参数让对象成为rea...

1895

扫码关注云+社区