1

我尝试编写一个程序,其中 Dictionary 由 List 索引。(相信我,是的,有选项,但我喜欢按列表索引)。有一个最小的工作(实际上不工作,只有最后一行是一个问题)示例:

using System;
using System.Collections.Generic;

namespace test
{
    class Program
    {
    static void Main(string[] args)
    {
        Dictionary<List<String>, int> h = new Dictionary<List<string>,int>();

        List<String> w = new List<string> {"a"};
        h.Add(w, 1);

        w = new List<string>{"b"};
        h.Add(w,2);

        w = new List<string>{"a"};

        int value = 0;
        h.TryGetValue(w, out value);
        Console.WriteLine(value+" "+h[w]);
    }
}

如果调试这个程序,他会清楚地看到 h 中有两个元素,但仍然无法通过正确的索引访问这些元素 --- h[w]。我错了还是发生了什么奇怪的事情?

4

4 回答 4

7

您的应用程序的问题源于以下事实:

new List<String> { "a" } != new List<String> { "a" }

列表的相等性检查两个引用是否引用同一个实例。在这种情况下,他们没有。相反,您创建了两个具有相同元素的列表......这并不能使它们相等。

您可以通过创建自定义平等比较器来解决此问题:

public class ListEqualityComparer<T> : IEqualityComparer<List<T>>
{
    public bool Equals(List<T> list1, List<T> list2)
    {
        return list1.SequenceEquals(list2);
    }

    public int GetHashCode(List<T> list)
    {
        if(list != null && list.Length > 0)
        {
            var hashcode = list[0].GetHashCode();
            for(var i = 1; i <= list.Length; i++)
                hashcode ^= list[i].GetHashCode();

            return hashcode;
        }

        return 0;
    }
}

然后将其传递给 Dictionary 构造函数:

Dictionary<List<String>, int> h = 
    new Dictionary<List<string>,int>(new ListEqualityComparer<String>());
于 2011-11-10T14:57:19.653 回答
2

问题是 List 的索引,您索引的不是列表中的数据,而是您本质上是通过指向 List 的内存指针(即该 List 所在的内存地址)进行索引。

您在一个内存位置创建了一个列表,然后在不同的内存位置创建了一个完全不同的列表(即当您创建一个新实例时)。这两个列表是不同的,即使它们包含相同的数据,这意味着您可以将任意数量的数据添加到字典中。

一种解决方案不是按列表索引而是按字符串索引并使用逗号分隔的列表,其中包含列表中的所有数据作为索引。

于 2011-11-10T14:56:53.443 回答
1

这对你永远不起作用,因为List<T>'sEqualsGetHashCode方法不考虑列表的内容。如果要使用对象集合作为键,则需要实现自己的集合类型,该集合类型Equals以检查集合中对象的相等性的方式进行覆盖(可能使用Enumerable.SequenceEqual.)

于 2011-11-10T14:57:33.327 回答
1

Dictionary 类使用引用比较来查找指定的键,这就是为什么即使列表包含相同的项目,它们也是不同的。

于 2011-11-10T14:57:38.693 回答