分享

c# – 带有IDictionary和IEnumerable的嵌套泛型

 印度阿三17 2019-06-10

在内部可重用库的通用C#类中,我想传递一个引用“映射到其他东西列表的东西”.那里传递的内容的数据类型不应该被库知道.此外,它们的存储方式也不应该是已知的,即今天的内存中保存的列表,以后可能是从按需读取的数据库表.

所以我以为我会写这个库类:

class GenericClass<T, U>
{
    public void Foo(IDictionary<T, IEnumerable<U>> bar)
    {
        // do something
    }
}

这会编译,但尝试传递具体实现不会:

class UsingClass
{
    public static void Main(string[] args)
    {
        var c = new GenericClass<string, string>();
        c.Foo(new Dictionary<string, List<string>>());
    }
}

我收到以下两个语法错误:

Filename.cs(46,13): error CS1502: The best overloaded method match for 'GenericClass<string,string>.Foo(System.Collections.Generic.IDictionary<string,System.Collections.Generic.IEnumerable<string>>)' has some invalid arguments
Filename.cs(46,19): error CS1503: Argument 1: cannot convert from 'System.Collections.Generic.Dictionary<string,System.Collections.Generic.List<string>>' to 'System.Collections.Generic.IDictionary<string,System.Collections.Generic.IEnumerable<string>>'

使用List替换Foo()声明中的IEnumerable会修复它,但这当然不是我想要的.

这真的不是C#(4.0)支持的,还是我错过了一些明显的东西?你会建议什么解决方法? (我确信之前已经讨论了很多,所以链接到很好的描述也很好.)

是的,我应该能够为此编写自己的帮助程序类,但为什么我必须这样做?

解决方法:

是的,这确实不受支持.想象一下你的Foo方法看起来像这样:

public void Foo(IDictionary<T, IEnumerable<U>> bar)
{
    T key = GetKeyFromSomewhere();
    bar[key] = new U[10]; // Create an array
}

看起来没关系,不是吗?我们可以从U []转换为IEnumerable< U>.

从调用者的角度来看,它并不是那么好 – 突然我们在字典中有一个string []引用值,当所有的值都是List< string>参考!邦去了类型安全.

您可以将方法重写为:

public void Foo<TValue>(IDictionary<T, TValue> bar)
    where TValue : IEnumerable<U>

这将让你从字典中获取值并将它们转换为IEnumerable< U>隐含地…但你只能将正确的值类型放入字典中,而你不能仅仅从U值构建它.

从版本4开始,C#支持在受限情况下的通用差异.例如,这适用于C#4(当针对.NET 4时)但以前不会:

List<string> strings = new List<string>();
IEnumerable<object> objects = strings;

有关泛型方差的更多信息,请参阅主题上的Eric Lippert’s blog series.做好准备让你的大脑定期爆炸.

来源:http://www./content-1-236201.html

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多