代码之家  ›  专栏  ›  技术社区  ›  chilltemp

如何在WCF上实现继承字典

  •  7
  • chilltemp  · 技术社区  · 16 年前

    我正在尝试实现一个用于WCF的字典。我的要求是:

    • 实际(私有变量或基 类)等同于 词典
    • 比较器 = System.StringComparer.InvariantCultureIgnoreCase
    • 自定义(覆盖/新建)添加(键, 价值)方法(包括 验证)。
    • 重写ToString()
    • 在客户端和主机上使用相同类型

    我尝试在WCF主机和客户端项目共享的公共项目中使用该类:

    [Serializable]
    public class MyDictionary : Dictionary<string, object>
    {
      public MyDictionary()
        : base(System.StringComparer.InvariantCultureIgnoreCase)
      { }
    
      public new void Add(string key, object value)
      { /* blah */ }
    
      public override string ToString()
      { /* blah */ }
    }
    
    [DataContract]
    [KnownType(typeof(MyDictionary))]
    [KnownType(typeof(object[]))]
    [KnownType(typeof(double[]))]
    [KnownType(typeof(string[]))]
    [KnownType(typeof(DateTime[]))]
    public class ResultClass
    {
      public object Value{ get; set; }
      /* More properties */
    }
    public class ParmData
    {
      public object Value{ get; set; }
      /* More properties */
    }
    [DataContract]
    [KnownType(typeof(MyDictionary))]
    [KnownType(typeof(object[]))]
    [KnownType(typeof(double[]))]
    [KnownType(typeof(string[]))]
    [KnownType(typeof(DateTime[]))]
    public class ParameterClass
    {
      public List<ParmData> Data{ get; set; }
      /* More properties */
    }
    
    [OperationContract]
    ResultClass DoSomething(ParameterClass args);
    

    结果:

    • 当我将myDictionary作为参数class.data.value元素之一传递时,会得到一个缺少的knownType异常。
    • 我可以在ResultClass中安全地返回我的字典,但它不再是我的类型。它只是一本字典,不能用来 MyDictionary . 比较器= System.Collections.Generic.GenericEqualityComparer<string> 不是我要找的不区分大小写的比较器。

    我需要的帮助是修复我失败的尝试,或者用一种完全不同的方式来达到我所述的要求。任何解决方案都不应涉及将一本词典复制到另一本词典。

    谢谢

    4 回复  |  直到 8 年前
        1
  •  10
  •   jezell    16 年前

    将CollectionDataContract添加到Dictionary类:

    有关使用收集数据协定实现字典的详细信息,请检查此链接:

    http://msdn.microsoft.com/en-us/library/aa347850.aspx

        2
  •  1
  •   Marc Gravell    16 年前

    前言:请注意,添加了一个“新的” Add 不停地打电话给老人 添加 只需铸造。而且,就“mex”而言,这是一个非常模糊的数据合同——是否需要如此开放?(许多) object 等等)

    第一:你没有错过一些吗 [DataContract] / [DataMember] 那里有标记吗?特别地:

    • 结果类值
    • 参数数据
    • PARAMDATA值
    • 参数类数据

    你能澄清吗? 确切地 您使用的是.NET的哪个版本? DataContractSerializer ETC已经通过服务包进行了调整。安装了3.5sp1(我唯一需要处理的事情)后,它至少可以通过 DataContractSerializer (没有wcf堆栈)和正确的 添加 方法被调用。

    您可以检查以下内容是否适用于您的本地版本吗?(它适用于3.5sp1和缺少的属性)【先输出】:

    1
    MyDictionary
    abc=123
    def=ghi
    

    代码:

            // or long-hand in C# 2.0
            ParameterClass pc = new ParameterClass {
                Data = new List<ParmData> { new ParmData {
                    Value = new MyDictionary  {
                        {"abc",123},
                        {"def","ghi"}
                    }}}};
            DataContractSerializer dcs = new DataContractSerializer(pc.GetType());
            string xml;
            using(StringWriter sw = new StringWriter())
            using(XmlWriter xw = XmlWriter.Create(sw)) {
                dcs.WriteObject(xw, pc);
                xw.Close();
                xml = sw.ToString();
            }
            using(StringReader sr = new StringReader(xml)) {
                ParameterClass clone = (ParameterClass)dcs.ReadObject(XmlReader.Create(sr));
                Console.WriteLine(clone.Data.Count);
                Console.WriteLine(clone.Data[0].Value.GetType().Name);
                MyDictionary d = (MyDictionary)clone.Data[0].Value;
                foreach (KeyValuePair<string, object> pair in d)
                {
                    Console.WriteLine("{0}={1}", pair.Key, pair.Value);
                }
            }
    

    显然,这只是测试 DataContractSerializer (没有整个WCF堆栈),但它似乎工作…那么:相同的代码在本地版本的.NET上工作吗?如果没有,是否可以安装最新的3.0 Service Pack?(最好安装3.5 SP1)。

    有关信息,我得到XML:

    <?xml version="1.0" encoding="utf-16"?><ParameterClass xmlns:i="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://schemas.datacontract.org/2004/07/"><Data><ParmData><Value xmlns:d4p1="http://schemas.microsoft.com/2003/10/Serialization/Arrays" i:type="d4p1:ArrayOfKeyValueOfstringanyType"><d4p1:KeyValueOfstringanyType><d4p1:Key>abc</d4p1:Key><d4p1:Value xmlns:d6p1="http://www.w3.org/2001/XMLSchema" i:type="d6p1:int">123</d4p1:Value></d4p1:KeyValueOfstringanyType><d4p1:KeyValueOfstringanyType><d4p1:Key>def</d4p1:Key><d4p1:Value xmlns:d6p1="http://www.w3.org/2001/XMLSchema" i:type="d6p1:string">ghi</d4p1:Value></d4p1:KeyValueOfstringanyType></Value></ParmData></Data></ParameterClass>
    
        3
  •  1
  •   chilltemp    16 年前
    • 使用 CollectionDataContract 杰泽尔建议的属性
    • 使用svcutil手动生成引用(代理)代码,使用/collectiontype参数。VS2008服务引用GUI不支持此参数。

    来源: WCF Collection Type Sharing

        4
  •  1
  •   lokusking    8 年前

    我终于找到了一个办法。我发现 this link 这是正确的方向,但我会尽量总结一下。

    1. 确保添加 [CollectionDataContract] 到您的自定义收藏集
    2. 像平常一样通过vs添加服务引用
    3. 展开服务引用并查找reference.svcmap文件
    4. 在“客户机选项”节点下,您将看到

    <集合映射/>

    将其替换为以下XML。

    <集合映射>
    <collectionmapping typename=“full.namespace.here”category=“list”/>
    </CollectionMappings>

    1. 右键单击服务引用,然后单击更新。它不应该再生成那个“代理”类,并且将允许您使用共享类。