如何将linq结果转换为HashSet或HashedSet

时间:2010-08-12 20:48:42

标签: c# linq

我在一个ISet类上有一个属性。我试图将linq查询的结果导入该属性,但无法弄清楚如何执行此操作。

基本上,寻找最后一部分:

ISet<T> foo = new HashedSet<T>();
foo = (from x in bar.Items select x).SOMETHING;

也可以这样做:

HashSet<T> foo = new HashSet<T>();
foo = (from x in bar.Items select x).SOMETHING;

9 个答案:

答案 0 :(得分:289)

我不认为有内置的东西可以做到这一点......但是编写扩展方法真的很容易:

public static class Extensions
{
    public static HashSet<T> ToHashSet<T>(
        this IEnumerable<T> source,
        IEqualityComparer<T> comparer = null)
    {
        return new HashSet<T>(source, comparer);
    }
}

请注意,您确实需要扩展方法(或至少是某种形式的通用方法),因为您可能无法明确表达T的类型:

var query = from i in Enumerable.Range(0, 10)
            select new { i, j = i + 1 };
var resultSet = query.ToHashSet();

通过显式调用HashSet<T>构造函数,您无法做到这一点。我们依靠通用方法的类型推断来为我们做这件事。

现在你可以选择将其命名为ToSet并返回ISet<T> - 但我会坚持使用ToHashSet和具体类型。这与标准LINQ运算符(ToDictionaryToList)一致,并允许将来扩展(例如ToSortedSet)。您可能还希望提供指定要使用的比较的重载。

答案 1 :(得分:67)

将IEnumerable传递给HashSet的构造函数。

HashSet<T> foo = new HashSet<T>(from x in bar.Items select x);

答案 2 :(得分:19)

答案 3 :(得分:17)

正如@Joel所说,你可以直接传递你的枚举。如果你想做一个扩展方法,你可以这样做:

public static HashSet<T> ToHashSet<T>(this IEnumerable<T> items)
{
    return new HashSet<T>(items);
}

答案 4 :(得分:3)

这很简单:)

var foo = new HashSet<T>(from x in bar.Items select x);

和是T是OP指定的类型:)

答案 5 :(得分:3)

如果您只需要只读访问该集合,而源代码是您方法的参数,那么我会选择

public static ISet<T> EnsureSet<T>(this IEnumerable<T> source)
{
    ISet<T> result = source as ISet<T>;
    if (result != null)
        return result;
    return new HashSet<T>(source);
}

原因是,用户可能已使用ISet调用您的方法,因此您无需创建副本。

答案 6 :(得分:2)

.NET框架和.NET核心中有一个扩展方法,用于将IEnumerable转换为HashSethttps://docs.microsoft.com/en-us/dotnet/api/?term=ToHashSet

public static System.Collections.Generic.HashSet<TSource> ToHashSet<TSource> (this System.Collections.Generic.IEnumerable<TSource> source);

似乎我还不能在.NET标准库中使用它(在撰写本文时)。那么我使用这个扩展方法:

    [Obsolete("In the .NET framework and in NET core this method is available, " +
              "however can't use it in .NET standard yet. When it's added, please remove this method")]
public static HashSet<T> ToHashSet<T>(this IEnumerable<T> source, IEqualityComparer<T> comparer = null) => new HashSet<T>(source, comparer);

答案 7 :(得分:1)

Jon的回答非常完美。唯一需要注意的是,使用NHibernate的HashedSet,我需要将结果转换为集合。有最佳方​​法吗?

ISet<string> bla = new HashedSet<string>((from b in strings select b).ToArray()); 

ISet<string> bla = new HashedSet<string>((from b in strings select b).ToList()); 

或者我错过了其他什么?


编辑:这就是我最终要做的事情:

public static HashSet<T> ToHashSet<T>(this IEnumerable<T> source)
{
    return new HashSet<T>(source);
}

public static HashedSet<T> ToHashedSet<T>(this IEnumerable<T> source)
{
    return new HashedSet<T>(source.ToHashSet());
}

答案 8 :(得分:0)

不是将IEnumerable简单转换为HashSet,而是将另一个对象的属性转换为HashSet通常很方便。你可以写成:

var set = myObject.Select(o => o.Name).ToHashSet();

但是,我的偏好是使用选择器:

var set = myObject.ToHashSet(o => o.Name);

他们做同样的事情,第二个显然更短,但我发现这个成语更适合我的大脑(我认为它就像ToDictionary)。

这是使用的扩展方法,支持自定义比较器作为奖励。

public static HashSet<TKey> ToHashSet<TSource, TKey>(
    this IEnumerable<TSource> source,
    Func<TSource, TKey> selector,
    IEqualityComparer<TKey> comparer = null)
{
    return new HashSet<TKey>(source.Select(selector), comparer);
}