Ø  简介

在 C# 中与数据交互最常用的语句就是 LINQ 了,而 LINQ to SQL 是最直接与数据库打交道的语句,它可以根据 LINQ 语法生成对应的 SQL 语句,在数据库中去执行。本文主要研究什么样 LINQ 语句会生成什么样 SQL 语句,从而确保编写 LINQ 语句时,便知道该语句会执行什么样的 SQL 语句,得到什么结果,这是很有必要的。因为很多时候会考虑到 SQL 效率问题,和数据是否有误等问题。主要包括:

1.   插入数据

2.   判断记录是否存在

3.   左连接情况下,判断某字段是否为 null 的写法

4.   连接查询关联不同类型(string & int)字段

5.   关联表(一对多关系),一条语句查询主表与子表数据

6.   使用 IQueryable<T> 类型的变量,嵌入 LINQ 查询语法中

7.   常见的查询技巧

8.   使用对象集合进行连接查询

1.   插入数据

1)   LINQ 语句

DataContext.CustomerTodos.Add(entity);

DataContext.SaveChanges();

2)   生成 SQL

exec 、49667的客户与用户记录

var dataList = (from t1 in DataContext.Customers

where t1.Id == 45357 || t1.Id == 49667

select new CustomerModel()

{

Id = (int)t1.Id,

Name = t1.Name,

UserList = (from d1 in DataContext.UserInfoes

where d1.CustomerId == t1.Id

select new CustomerModel.UserInfoModel()

{

Id = d1.id,

UserName = d1.userName

}).ToList()

}).ToList();

2)   生成 SQL

SELECT

3)   执行结果

4)   分析

1.   首先,我们定义了一个 IQueryable<T> 类型的变量,然后在使用这个变量作为数据源,进行关联查询。可见,在生成的 SQL 中直接进行了内连接查询,并输出相关字段。

2.   这种写法主要是可以方便多次查询,可以多次利用这个 allCustomers 条SQL)

exec

3.   分析

1)   每次遍历,都会生成不同的SQL,并查询数据库。

2)   通常情况下,不建议这样去遍历查询,可以先查询出所需的记录,再进行遍历。

3)   根据不同条件生成对应 SQL

1.   LINQ 语句

var query3 = (from t1 in dbContext.Grades

where t1.GradeName.Contains("年级")

orderby t1.GradeId descending

select t1);

for (int i = 1; i <= 3; i++)

{

var entity = query3.Where(o => o.GradeId == i).FirstOrDefault();

Console.WriteLine("获取记录:{0}", (entity != null ? entity.GradeId : 0));}

2.   生成 SQL(类似的3条SQL)

exec sp_executesql N'SELECT TOP (1)

[Project1].[GradeId] AS [GradeId],

[Project1].[GradeName] AS [GradeName],

[Project1].[Remark] AS [Remark]

FROM ( SELECT

[Extent1].[GradeId] AS [GradeId],

[Extent1].[GradeName] AS [GradeName],

[Extent1].[Remark] AS [Remark]

FROM [dbo].[Grade] AS [Extent1]

WHERE ([Extent1].[GradeName] LIKE N''%年级%'') AND ([Extent1].[GradeId] = @p__linq__0) AND (@p__linq__0 IS NOT NULL)

)  AS [Project1]

ORDER BY [Project1].[GradeId] DESC',N'@p__linq__0 int',@p__linq__0=1

3.   分析

1)   每次遍历,都会生成不同的SQL,并查询数据库。

2)   并将条件以 AND(与)进行追加。

3)   通常情况下,不建议这样去遍历查询,可以先查询出所需的记录,再进行遍历。

8.   使用对象集合进行连接查询

1)   LINQ 语句

Class1[] objects = new Class1[]

{

new Class1 { Id = 1, Name = "Name1" },

new Class1 { Id = 2, Name = "Name2" },

new Class1 { Id = 3, Name = "Name3" }

};

var query1 = (from t1 in objects

join t3 in dbContext.Grades on t1.Id equals t3.GradeId

where t1.Name.Contains("2") && t3.GradeName.Contains("二")

select new { t3.GradeId, t1.Name, t3.GradeName });

foreach (var item in query1)

{

Console.WriteLine("获取记录:{0},{1},{2}", item.GradeId, item.Name, item.GradeName);

}

2)   生成 SQL

SELECT

[Extent1].[GradeId] AS [GradeId],

[Extent1].[GradeName] AS [GradeName],

[Extent1].[Remark] AS [Remark]

FROM [dbo].[Grade] AS [Extent1]

3)   执行结果

4)   分析

1.   可见,以对象集合关联查询,生成的查询 SQL 中并没有任何 where 条件。

2.   首先查询所有的“右表数据”,再在程序中进行过滤。

3.   注意:这种关联查询并不是一种标准查询,并且对象集合必须为左表(t1的位置)。否则将抛出异常:System.NotSupportedException,无法创建“EFDBFirst6_0.Class1”类型的常量值。此上下文仅支持基元类型或枚举类型。

4.   最后,不推荐使用该方式连接查询,影响正常思维和效率。

深入理解 LINQ to SQL 生成的 SQL 语句的更多相关文章

  1. 查看Linq to Sql生成的sql语句(转)

    查看Linq to Sql生成的sql语句   在控制台项目中,比较简单,直接db.Log = Console.Out;就OK了 但是在其他项目中,需要处理如下: StreamWriter sw = ...

  2. SqlProfilter监控指定数据库数据表——监控linq组合查询生成的sql

    1.例子 实际测试环境中往往很多测试都在调用数据库,那么如何使用SqlProfilter监控筛选到自己想要监看的数据库对应的表有关linq生成的sql时候就需要做如下设置了 ........... u ...

  3. netcore sqlserver linq contains生成的sql语句不是使用like而是charIndex

    在ef中使用linq调用了contains,结果怎么都查不到值,打开sqlserver profiler 发现生成的sql语句不是使用like...而是CharIndex 参考文档:https://s ...

  4. Linq To EF 用泛型时生成的Sql会查询全表的问题

    1.问题的现象 public class LinqHepler<T> where T:class { private EFDBContext _context = null; /// &l ...

  5. 简单理解ORM,实体类生成查询SQL语句

    目前有很多开源的ORM项目,大多情况下也不需要我们重复去造轮子,我们只需要了解轮子怎么造的,怎么用就可以,下面简单说一下怎么通过实体生成一个SQL语句: 先建立2个Attribute类,TableAt ...

  6. Rafy 领域实体框架设计 - 重构 ORM 中的 Sql 生成

    前言 Rafy 领域实体框架作为一个使用领域驱动设计作为指导思想的开发框架,必然要处理领域实体到数据库表之间的映射,即包含了 ORM 的功能.由于在 09 年最初设计时,ORM 部分的设计并不是最重要 ...

  7. 用LinqPad查看Nhibernate生成的sql语句

    使用Nhibernate开发一般都要对Nhibernate生成的sql语句进行查看及分析,查看Nhibernate生成的sql语句,可以使用NHProfiler和log4net.但NHProfiler ...

  8. 使用工具追踪Entity Framework生成的SQL

    学习entity framework期间收集的文章,转自http://www.cnblogs.com/hiteddy/archive/2011/10/01/Difference_among_IQuer ...

  9. EF查询生成的SQL

    在EF 4和EF 3.5 SP1中,我们可以使用ToTraceString()方法得到EF查询所生成的SQL. using (var context = new TestDBEntities()) { ...

随机推荐

  1. css_选择器

    老师的博客:https://www.cnblogs.com/liwenzhou/p/7999532.html 参考w3 school:http://www.w3school.com.cn/css/cs ...

  2. java程序启动 环境属性的获取

    System.getProperties().list(System.out); 如果要获取某一个属性,例如常见的“操作系统” 则  System.getProperty("os.name& ...

  3. w3m 使用总结

    安装 sudo apt install w3m终端 w3m www.baidu.com 即可打开w3m是个开放源代码的命令行下面的网页浏览器.一般的linux系统都会自带这个工具,可以通过它在命令行下 ...

  4. 利用XShell上传、下载文件(使用sz与rz命令)

    XSHELL工具上传文件到Linux以及下载文件到本地(Windows)   Xshell很好用,然后有时候想在windows和linux上传或下载某个文件,其实有个很简单的方法就是rz,sz.首先你 ...

  5. 2、FreeRTOS任务相关API函数

    1.任务相关的API函数 函数存在于task.c中,主要的函数有: xTaskCreate():使用动态的方法创建一个任务: xTaskCreatStatic():使用静态的方法创建一个任务(用的非常 ...

  6. Label Encoding vs One Hot Encoding

    最近在刷kaggle的时候碰到了两种处理类别型特征的方法:label encoding和one hot encoding.我从stackexchange, quora等网上搜索了相关的问题,总结如下. ...

  7. Codeforces Round #546 (Div. 2)-D - Nastya Is Buying Lunch

    这道题,神仙贪心题... 题意就是我给出数的顺序,并给出多个交换,每个只能用于相邻交换,问最后一个元素,最多能往前交换多少步. 我们考虑这样一个问题,如果一个这数和a[n]发生交换,那么这个数作为后面 ...

  8. python 必学模块collections

    包含的主要功能如下 查看collections 的源码我们可以看到其为我们封装了以下的数据结果供我们调用 __all__ = ['deque', 'defaultdict', 'namedtuple' ...

  9. Shell脚本中的break continue exit return

    转自:http://www.cnblogs.com/guosj/p/4571239.html break结束并退出循环 continue在循环中不执行continue下面的代码,转而进入下一轮循环 e ...

  10. java内存模型详解

    对于本篇文章,将从四个概念来介绍:内存模型基础,重排序,顺序一致性和happens-before 1.内存模型基础 在并发编程中,有两个关键问题:线程之间如何通信和如何同步.由此而引出了两种并发模型: ...