linq group join

本篇介绍Linq的Group和Join操作,继续使用《Linq 学习(3) 语法结构》中介绍的数据源。

Group
Group是进行分组操作,同SQL中的Group By类似。
原型如下:

public static IEnumerable<IGrouping<TKey, TSource>> GroupBy<TSource, TKey>( 
    this IEnumerable<TSource> source, 
    Func<TSource, TKey> keySelector)

它有几个重载,返回类型有两种:IEnumerable<IGrouping<TKey, TSource>> 和 IEnumerable<TResult>。
返回类型为:IEnumerable<IGrouping<TKey, TSource>>
示例:
返回按学号分组学生的成绩

var result = from score in DataSource.Scores 
             group score by score.StudentID into scoreGroup 
             select scoreGroup;

scoreGroup为IGrouping<TKey, TSource>类型,返回结果为IEnumerable<IGrouping<TKey, TSource>>,既集合的集合,因此输出时需用双重循环。
IGrouping<TKey, TElement>接口定义为:

public interface IGrouping<TKey, TElement> : IEnumerable<TElement>, IEnumerable 

    TKey Key { get; } 
}

其中Key为分组依据的字段。

foreach (var group in result) 

    //输出分组依据的字段 
    Console.WriteLine("\nStudent ID:" + group.Key);

// 输出组内成员 
    foreach (var score in group) 
    { 
        Console.WriteLine(score); 
    } 
}

// result: 
// Student ID:1 
// Student ID:1,Course ID:1,Score:78 
// Student ID:1,Course ID:2,Score:60 
// ...

// Student ID:2 
// Student ID:2,Course ID:1,Score:59 
// ...

等效的扩展方法调用实现为:

var result = DataSource.Scores.GroupBy(score => score.StudentID);

返回类型为:IEnumerable<TResult>
对分组结果进行一些包装,如包装为匿名类型。
返回按学号分组学生的成绩

var result = from score in DataSource.Scores 
             group score by score.StudentID into scoreGroup 
             select new { StudentID = scoreGroup.Key, Group = scoreGroup };

匿名类型中Group为IGrouping<TKey, TSource>类型。
等效的扩展方法调用实现为:

var result = DataSource.Scores.GroupBy(score => score.StudentID, 
    (key, group) => new { StudentID = key, Group = group });

其他一些重载使用方法类似。

Join

连接操作。

public static IEnumerable<TResult> Join<TOuter, TInner, TKey, TResult>( 
    this IEnumerable<TOuter> outer, 
    IEnumerable<TInner> inner, 
    Func<TOuter, TKey> outerKeySelector, 
    Func<TInner, TKey> innerKeySelector, 
    Func<TOuter, TInner, TResult> resultSelector)

从Join方法原型可以看出其使用方法。

内连接
选择左右两侧集合都含有相对应的元素。
示例:
查询学生的姓名、学科、成绩。

var result = from score in DataSource.Scores 
             join student in DataSource.Students on score.StudentID equals student.StudentID 
             join course in DataSource.Courses on score.CourseID equals course.CourseID 
             select new { StudentName = student.Name, CourseName = course.CourseName, ScoreValue = score.Value }; 
// result 
// { StudentName = Andy, CourseName = C Language, ScoreValue = 78 } 
// { StudentName = Andy, CourseName = Biophysics, ScoreValue = 60 } 
// ... 
// { StudentName = Bill, CourseName = C Language, ScoreValue = 59 } 
// { StudentName = Cindy, CourseName = Biophysics, ScoreValue = 60 } 
// ...

等效的扩展方法调用实现为:

var result = 
    DataSource.Scores.Join( 
    DataSource.Students, 
    score => score.StudentID, 
    student => student.StudentID, 
    (score, student) => new { StudentName = student.StudentID, ScoreValue = score.Value, CourseID = score.CourseID }) 
    .Join(DataSource.Courses, 
    scostu => scostu.CourseID, 
    course => course.CourseID, 
    (scostu, course) => new { StudentName = scostu.StudentName, CourseName = course.CourseName, ScoreValue = scostu.ScoreValue });

左外连接
当右侧的连接的右侧没有左侧对应的元素时,内连接会忽略左侧元素。要想保留左侧元素,可以使用做外连接。右侧被置为默认值,如:引用类型被置为空。
示例:

var result = 
    from student in DataSource.Students2 
    join score in DataSource.Scores on student.StudentID equals score.StudentID into Scores 
    from score in Scores.DefaultIfEmpty() 
    select new { student = student, score = score == default(Score) ? 0 : score.Value }; 
// result: 
// { student = Student ID:5,Student Name:Erik, score = 78 } 
// { student = Student ID:6,Student Name:Frank, score = 0 }

等效的扩展方法调用实现为: 
var result = 
    DataSource.Students2.GroupJoin( 
    DataSource.Scores, 
    student => student.StudentID, 
    score => score.StudentID, 
    (student, Scores) => new { student = student, Scores = Scores }) 
    .SelectMany(group => group.Scores.DefaultIfEmpty(), 
    (group, score) => new { student = group.student, score = (score == null) ? 0.0 : score.Value });

笛卡尔积
集合中的元素交错连接。
示例:统计学生课程成绩时的模板。

var result = from student in DataSource.Students 
             from course in DataSource.Courses 
             select new { StudentName = student.Name, CourseName = course.CourseName, ScoreValue = (double?)null }; 
// result: 
// { StudentName = Andy, CourseName = C Language, ScoreValue =  } 
// { StudentName = Andy, CourseName = Biophysics, ScoreValue =  } 
// ... 
// { StudentName = Bill, CourseName = C Language, ScoreValue =  } 
// ... 
// { StudentName = Cindy, CourseName = Fundamentals of Compiling, ScoreValue =  } 
// ...

等效的扩展方法调用实现为: 
var result = DataSource.Students.SelectMany( 
    student => DataSource.Courses 
        .Select( 
        course => 
            new { StudentName = student.Name, CourseName = course.CourseName, ScoreValue = (double?)null }));

GroupJoin
连接分组。
方法原型为:

public static IEnumerable<TResult> GroupJoin<TOuter, TInner, TKey, TResult>( 
    this IEnumerable<TOuter> outer, 
    IEnumerable<TInner> inner, 
    Func<TOuter, TKey> outerKeySelector, 
    Func<TInner, TKey> innerKeySelector, 
    Func<TOuter, IEnumerable<TInner>, TResult> resultSelector)

// result: 
// Andy 
// 1----78 
// 2----60 
// ... 
// Bill 
// 1----59 
// ... 
// Cindy 
// 2----60 
// ...

相当于组合了Group操作和Join操作。等效的操作如下:

var result = from item in 
                 (from student in DataSource.Students 
                  join score in DataSource.Scores on student.StudentID equals score.StudentID 
                  select new { StudentName = student.Name, CourseID = score.CourseID, Value = score.Value }) 
             group item by item.StudentName into Group 
             select new { StudentName = Group.Key, Group = Group };

结束语
到现在,Linq与SQL语言等价的操作基本介绍完,组合这些操作能实现复杂的查询。

时间: 2024-10-27 14:15:33

linq group join的相关文章

Linq中join &amp; group join &amp; left join 的用法

Linq中join & group join & left join 的用法 2013-01-30 11:12 12154人阅读 评论(0) 收藏 举报  分类: C#(14)  文章转自:http://www.cnblogs.com/c-jquery-linq-sql-net-problem/archive/2011/01/17/LINQ_Inner_Join_Group_Join_Left_Join.html 我们在做SQL查询的时候经常会用到Inner Join,Left Join,

Linq之Join操作

1 摘要 文章通过一个简单的实例对Linq中的Join操作进行演示,并在文章的最后对Join操作相关知识点进行简单的总结. 2 实例演示 1) 新建数据库MyTestDB,在数据库中新建数据表tb_Class和tb_Student,两表的定义如下图所示.                                        图1  tb_Class的定义                                                                    

EF INNER JOIN,LEFT JOIN,GROUP JOIN

IQueryable<TOuter>的扩展方法中提供了 INNER JOIN,GROUP JOIN但是没有提供LEFT JOIN GROUP JOIN适用于一对多的场景,如果关联的GROUP没有条目,会显示List条目为0,这一点其实也是LEFT join, 但是如果反过来,对于多对一的场景,虽然可以用GROUP JOIN,但是对于单一的条目却还要用List进行包装,就有点逻辑的冗余. 这个时候Left join就派上用场了 /// <summary> /// InnerJoin

Linq 中 Join 的用法

Linq中连接主要有组连接.内连接.左外连接.交叉连接四种.各个用法如下. 注:本文内容主要来自<Linq实战>,本例中用到的对象请见文章底部. 1. 组连接 组连接是与分组查询是一样的.即根据分组得到结果. 如下例,根据publisther分组得到结果. 使用组连接的查询语句如下: //使用组连接            var GroupQuery = from publisher in SampleData.Publishers                             jo

linq group by max 多表链接实例

SELECT s.* FROM dbo.ERG_TipOffsInfo s, (SELECT Data,MAX(Createtime) max_Time FROM dbo.ERG_TipOffsInfo GROUP BY Data) t WHERE s.Data=t.Data AND s.CreateTime = t.max_Time AND s.TipOffsTypeId=2 ORDER BY s.Createtime DESC linq 改写代码 var list = from s in c

C# Linq group by 和 group by into 运用实例

Form:在一个查询中如果需要查询多个数据源则使用多个From Group:分组,将查询结果按照指定分组条件放入到不同的分组中 Into:使用Into上下文关键字创建一个临时标示符,一遍将group.join或select子句的结果存储到新的标示符中 Let:在查询中增加一个范围变量,可以直接运算来的,也可以调用外部方法来存放外部方法得到的结果集: List<Student> listStudent = new List<Student>            {         

LINQ学习——JOIN

一.JOIN的作用 1.使用联接来结合两个或更多的集合的数据. 2.联接操作接受两个集合然后创建一个临时的对象集合,每一个对象包含原始集合对象中的所有字段. Note:这里是包含而不是这个原实集合的字段一定要使用,这要看SELECT原始集合的哪些字段. 二.LINQ表达式的语法 Jion Identifier in Collection2 On Field1 equals Field2      Note:使用上下文关键字"equals"来比较字段,不能用"=="这

LINQ Group By操作

假设我们需要从两张表中统计出热门商圈,这两张表内容如下: 上表是所有政区,商圈中的餐饮个数,名为FoodDistrict 下表是所有政区,商圈中的SPA个数,名为SPADistrict 现在要把这两张表,根据政区和商圈合并,然后相加Counts,根据Counts的总大小排序,统计热门商圈和热门政区. 在这里仅讨论合并的问题,以演示在SQLServer和C#中LINQ的实现方法: 通常,我们可以直接通过在SQLServer里面首先通过Union All,然后再通过GroupBy语句来执行查询操作即

linq left join ,inner join ,crossjoin

inner join : linq 默认使用Inner Join的链接方式,如下面的表达式一样: Left Join: 左链接返回左表的全部数据,以及右表中满足链接条件和不满足链接条件的数据,不满足的取字段值, 在一对多的的情况下主表的数据会被重复, 左外链接的实现方式如下,注意这里面和上面的写法不同点在于,链接之后做了一个Into n,n这个时候就是一个SalesOrderDetails的列表, 如果你想返回SalesOrderDetails的数据,你需要再增加一步 from b2 in n.