LINQ选择不同的集合并检索不同的列

时间:2015-01-14 21:54:32

标签: c# linq

我希望能够执行以下查询:

select distinct studentname, subject, grade from studenttable;

学生表包含以下字段:

studentid
studentname
subject
grade

我现在拥有的linq查询是:

var students=dc.Select(s => new {s.studentname, s.subject, s.grade}).Distinct();

dc是数据上下文。

此查询有效,但如何让学生id同时仍然满足studentname, subject, grade集上的不同条件?

3 个答案:

答案 0 :(得分:4)

这里的问题是您已经收集了必要数据的三个属性,虽然它们足以通过Distinct,但没有任何链接回原始数据,并且任何此类链接都会破坏Distinct的默认实现。

但是,您可以使用不同which takes an IEqualityComparer的重载。这将允许您仅在所需字段上进行相等性比较,同时在整个集合上运行Distinct。

var students = dc
    .AsEnumerable() // In case you're using a linq-to-sql framework, this will ensure the query execute in-memory
    .Distinct(new SpecialtyComparer());

//...

public class SpecialtyComparer : IEqualityComparer<StudentTable>
{
    public int GetHashCode(StudentTable s)
    {
        return s.studentname.GetHashCode()
            && s.subject.GetHashCode()
            && s.grade.GetHashCode();
    }

    public bool Equals(StudentTable s1, StudenTable s2)
    {
        return s1.studentname.Equals(s2.studentname)
            && s1.subject.Equals(s2.subject)
            && s1.grade.Equals(s2.grade);
    }
}

答案 1 :(得分:2)

我相信你的设计已被破坏,但我会回答你的具体问题......

我假设您尝试按名称,科目和成绩进行分组,并检索每个小组的第一位代表学生。

在这种情况下,您可以按元组进行分组。元组将免费提供EqualsGetHashCode方法,因此可以在组操作中使用。

IEnumerable<Student> distinctStudents = students
                                           .AsEnumerable()
                                           .GroupBy(s => Tuple.Create
                                                               (
                                                                   s.studentname, 
                                                                   s.subject, 
                                                                   s.grade
                                                               )
                                                          )
                                           .Select(g => g.First()); /// Returns the first student of each group

这是一个点网小提琴示例: https://dotnetfiddle.net/7K13DJ

答案 2 :(得分:1)

在对象列表上执行Distinct()时,您将这些行聚合为较少的行,并丢弃任何重复行。因此,您的结果将不再具有studentid。要保留studentid属性,您需要使用GroupBy。这将返回您的密钥(studentnamesubjectgrade的学生)以及原始行列表。您可以在此处查看此问题中的示例:(GroupBy and count the unique elements in a List)。

您将有一个studentids列表供您选择,因为可能有许多行具有相同的studentnamesubjectgrade。 (否则你不会做一个独特的 - 它们将是唯一的,tuple { sudentname, subject, grade }将是自然键)。所以你可能需要问自己的问题是“我需要哪种学生?”