LINQ系列:LINQ to SQL Join连接

本文深入探讨了SQL中的一对多查询、双向关联、左外连接和内联多个表的高级查询技巧,通过具体代码实例展示如何进行复杂的数据关联与筛选。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1. 一对多

var expr = context.Products
    .Where(p => p.Category.CategoryName == "LINQ to SQL" && p.UnitPrice > 10m)
    .Select(p => new
    {
        p.ProductID,
        p.ProductName
    });
var expr = from p in context.Products
            where p.Category.CategoryName == "LINQ to SQL" && p.UnitPrice > 10m
            select new
            {
                p.ProductID,
                p.ProductName
            };
SELECT 
    [Extent1].[ProductID] AS [ProductID], 
    [Extent1].[ProductName] AS [ProductName]
    FROM  [dbo].[Product] AS [Extent1]
    INNER JOIN [dbo].[Category] AS [Extent2] ON [Extent1].[CategoryID] = [Extent2].[CategoryID]
    WHERE (N'LINQ to SQL' = [Extent2].[CategoryName]) AND ([Extent1].[UnitPrice] > cast(10 as decimal(18)))
var expr = from p in context.Products
            where p.UnitPrice > 10m
            join c in context.Categories on p.CategoryID equals c.CategoryID
            select new
            {
                p.ProductID,
                p.ProductName,
                c.CategoryName
            };
SELECT 
    [Extent1].[ProductID] AS [ProductID], 
    [Extent1].[ProductName] AS [ProductName], 
    [Extent2].[CategoryName] AS [CategoryName]
    FROM  [dbo].[Product] AS [Extent1]
    INNER JOIN [dbo].[Category] AS [Extent2] ON [Extent1].[CategoryID] = [Extent2].[CategoryID]
    WHERE [Extent1].[UnitPrice] > cast(10 as decimal(18))

 

var expr = from p in context.Products
           join c in context.Categories on p.CategoryID equals c.CategoryID
           select new
           {
               p.ProductID,
               p.ProductName,
               c.CategoryName
           };

 

SELECT 
    [Extent1].[ProductID] AS [ProductID], 
    [Extent1].[ProductName] AS [ProductName], 
    [Extent2].[CategoryName] AS [CategoryName]
    FROM  [dbo].[Product] AS [Extent1]
    INNER JOIN [dbo].[Category] AS [Extent2] ON [Extent1].[CategoryID] = [Extent2].[CategoryID]

2. 双向关联

var expr = from c in context.Categories
            join p in context.Products on c.CategoryID equals p.CategoryID into cp
            select new
            {
                c.CategoryName,
                TotalCategoryProducts = cp.Count()
            };
SELECT 
    [Extent1].[CategoryID] AS [CategoryID], 
    [Extent1].[CategoryName] AS [CategoryName], 
    (SELECT 
        COUNT(1) AS [A1]
        FROM [dbo].[Product] AS [Extent2]
        WHERE [Extent1].[CategoryID] = [Extent2].[CategoryID]) AS [C1]
    FROM [dbo].[Category] AS [Extent1]

3. left out join

var expr = from c in context.Categories
            join p in context.Products on c.CategoryID equals p.CategoryID into CategoryProducts
            from cp in CategoryProducts.DefaultIfEmpty()
            select new
            {
                c.CategoryID,
                c.CategoryName,
                TotalProducts = CategoryProducts.Count()
            };
SELECT 
    [Extent1].[CategoryID] AS [CategoryID], 
    [Extent1].[CategoryName] AS [CategoryName], 
    (SELECT 
        COUNT(1) AS [A1]
        FROM [dbo].[Product] AS [Extent3]
        WHERE [Extent1].[CategoryID] = [Extent3].[CategoryID]) AS [C1]
    FROM  [dbo].[Category] AS [Extent1]
    LEFT OUTER JOIN [dbo].[Product] AS [Extent2] ON [Extent1].[CategoryID] = [Extent2].[CategoryID]

4. inner join select many

var expr = from c in context.Categories
           from p in c.Products
           where c.CategoryName == "LINQ to SQL"
           select p;
var expr = context.Categories
                .Where(c => c.CategoryName == "LINQ to SQL")
                .SelectMany(c => c.Products);
SELECT 
    [Extent2].[ProductID] AS [ProductID], 
    [Extent2].[CategoryID] AS [CategoryID], 
    [Extent2].[ProductName] AS [ProductName], 
    [Extent2].[UnitPrice] AS [UnitPrice], 
    [Extent2].[UnitsInStock] AS [UnitsInStock], 
    [Extent2].[Discontinued] AS [Discontinued]
    FROM  [dbo].[Category] AS [Extent1]
    INNER JOIN [dbo].[Product] AS [Extent2] ON [Extent1].[CategoryID] = [Extent2].[CategoryID]
    WHERE N'LINQ to SQL' = [Extent1].[CategoryName]
Join操作 适用场景:在我们关系中有一对一关系,一对多关系,多对多关系等。对各个之间的关系,就用这些实现对多个的操作。 说明:在Join操作中,分别为Join(Join查询), SelectMany(Select一对多选择)和GroupJoin(分组Join查询)。 该扩展方法对两个序列中键匹配的元素进行inner join操作 SelectMany 说明:我们在写查询语句时,如果被翻译成SelectMany需要满足2个条件。1:查询语句中没有join和into,2:必须出现EntitySet。在我们关系中有一对一关系,一对多关系,多对多关系等,下面分别介绍一下。 1.一对多关系(1 to Many): var q = from c in db.Customers from o in c.Orders where c.City == "London" select o; 语句描述:Customers与Orders是一对多关系。即Orders在Customers类中以EntitySet形式出现。所以第二个 from是从c.Orders而不是db.Orders里进行筛选。这个例子在From子句中使用外键导航选择伦敦客户的所有订单。 var q = from p in db.Products where p.Supplier.Country == "USA" && p.UnitsInStock == 0 select p; 语句描述:这一句使用了p.Supplier.Country条件,间接关联了Supplier。这个例子在Where子句中使用外键导航筛选其供应商在美国且缺货的产品。生成SQL语句为: SELECT [t0].[ProductID], [t0].[ProductName], [t0].[SupplierID], [t0].[CategoryID],[t0].[QuantityPerUnit],[t0].[UnitPrice], [t0].[UnitsInStock], [t0].[UnitsOnOrder],[t0].[ReorderLevel], [t0].[Discontinued] FROM [dbo].[Products] AS [t0] LEFT OUTER JOIN [dbo].[Suppliers] AS [t1] ON [t1].[SupplierID] = [t0].[SupplierID] WHERE ([t1].[Country] = @p0) AND ([t0].[UnitsInStock] = @p1) -- @p0: Input NVarChar (Size = 3; Prec = 0; Scale = 0) [USA] -- @p1: Input Int (Size = 0; Prec = 0; Scale = 0) [0] 2.多对多关系(Many to Many): var q = from e in db.Employees from et in e.EmployeeTerritories where e.City == "Seattle" select new { e.FirstName, e.LastName, et.Territory.TerritoryDescription }; 说明:多对多关系一般会涉及三个(如果有一个是自关联的,那有可能只有2个)。这一句语句涉及Employees, EmployeeTerritories, Territories三个。它们的关系是1:M:1。Employees和Territories没有很明确的关系。 LINQ to SQL语句之Join和Order By部分代码 语句描述:这个例子在From子句中使用外键导航筛选在西雅图的雇员,同时列出其所在地区。这条生成SQL语句为: SELECT [t0].[FirstName], [t0].[LastName], [t2].[TerritoryDescription] FROM [dbo].[Employees] AS [t0] CROSS JOIN [dbo].[EmployeeTerritories] AS [t1] INNER JOIN [dbo].[Territories] AS [t2] ON [t2].[TerritoryID] = [t1].[TerritoryID] WHERE ([t0].[City] = @p0) AND ([t1].[EmployeeID] = [t0].[EmployeeID]) -- @p0: Input NVarChar (Size = 7; Prec = 0; Scale = 0) [Seattle] 。。。。。。。。。。。。。。。。。。。。。。。。。。。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值