我希望能够执行以下查询:
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
集上的不同条件?
答案 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)
我相信你的设计已被破坏,但我会回答你的具体问题......
我假设您尝试按名称,科目和成绩进行分组,并检索每个小组的第一位代表学生。
在这种情况下,您可以按元组进行分组。元组将免费提供Equals
和GetHashCode
方法,因此可以在组操作中使用。
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
。这将返回您的密钥(studentname
,subject
,grade
的学生)以及原始行列表。您可以在此处查看此问题中的示例:(GroupBy and count the unique elements in a List)。
您将有一个studentids
列表供您选择,因为可能有许多行具有相同的studentname
,subject
和grade
。 (否则你不会做一个独特的 - 它们将是唯一的,tuple { sudentname, subject, grade }
将是自然键)。所以你可能需要问自己的问题是“我需要哪种学生?”