asp.net core系列 33 EF查询数据 (2)
一. 原生sql查询
接着上篇讲。通过 entity framework core 可以在使用关系数据库时下降到原始 sql 查询。 在无法使用 linq 表达要执行的查询时,或因使用 linq 查询而导致低效的 sql 查询时非常有用。 原始 sql 查询可返回实体类型,或者从 ef core 2.1 开始,可返回模型中的查询类型。
1.1 基本的原始sql查询
可以使用fromsql扩展方法,基于原始的sql查询,开始linq查询。
var blogs = context.blogs .fromsql("select * from dbo.blogs") .tolist();
--通过sql server profiler监听的sql,如下所示: select * from dbo.blogs
1.2 原生 sql 查询可用于执行存储过程(getmostpopularblogs)
var blogs = context.blogs .fromsql("execute dbo.getmostpopularblogs") .tolist();
1.3 传递参数
在使用fromsql执行原始sql查询时,传递参数要防止sql注入攻击,可以在sql查询字符串中包含参数占位符,然后提供参数值作为附加参数。任何参数值将自动转换为dbparameter来防止sql注入。
var id = 4; var blgos= bloggingcontext.blogs.fromsql("select * from dbo.blogs where blogid={0}",id).tolist();
--通过sql server profiler监听的sql,如下所示: exec sp_executesql n'select * from dbo.blogs where blogid=@p0',n'@p0 int',@p0=4
可以构造 dbparameter 并将其作为参数值提供。 这样可以在 sql 查询字符串中使用命名参数(与上面占位符实现一样,只不过这里显示提供了命名参数@blogid)。
var user = new sqlparameter("@blogid", 4); var blgos= bloggingcontext.blogs.fromsql("select * from dbo.blogs where blogid={0}", user).tolist();
--通过sql server profiler监听的sql,如下所示: exec sp_executesql n'select * from dbo.blogs where blogid=@blogid',n'@blogid int',@blogid=4
1.4 使用 linq 编写
发送到数据库中的 sql 查询可以是组合的,则可以在原始 sql 查询后面紧跟着使用 linq 运算符。 以 select
关键字开始的 sql 查询一般是可组合的。以下示例使用原始sql查询,然后使用linq对其进行编写以执行过滤和排序。
var blgos= bloggingcontext.blogs .fromsql("select blogid,url from dbo.blogs") .where(b=>b.blogid>2) .orderbydescending(b=>b.blogid) .select(b=> new{ b.blogid, b.url}) .tolist();
--通过sql server profiler监听的sql,原始sql成了一个子查询,如下所示: select [b].[blogid], [b].[url] from ( select blogid,url from dbo.blogs ) as [b] where [b].[blogid] > 2 order by [b].[blogid] desc
二. 异步查询
当在数据库中执行查询时,异步查询可避免阻止线程。 这有助于避免冻结富客户端应用程序的 ui。 异步操作还可以增加 web 应用程序的吞吐量,可以在等数据库操作完成时(i/o),释放当前线程到线程池,该线程可去处理其他请求。
注意: ef core 不支持在同一上下文实例上运行多个并行操作。 应始终等待操作完成,然后再开始下一个操作。 这通常是通过在每个异步操作上使用 await 关键字完成的。
entity framework core 提供了一组异步扩展方法,可用作执行查询并返回结果的 linq 方法的替代方法。示例包括 tolistasync()
、toarrayasync()
、singleasync()
等。对于部分 linq 运算符(如 where(...)
、orderby(...)
等),没有对应的异步版本,因为这些方法仅用于构建 linq 表达式树,而未将查询发送到数据库中执行。
下面是一个示例,注意async必须搭配await,只有await完成后才会释放当前ef上下文, async后面必须是task(无返回值)或task<t>(有返回值):
public async task<list<blog>> getblogsasync() {return await context.blogs.tolistasync(); }
三. 全局查询筛选器
全局查询筛选器是应用于元数据模型(通常为 onmodelcreating)中的实体类型的 linq 查询谓词(通常传递给 linq where 查询运算符的布尔表达式)。
下面使用 hasqueryfilter
api 在 onmodelcreating 中配置查询筛选器。
protected override void onmodelcreating(modelbuilder modelbuilder) { //isdeleted==1 没有被逻辑删除的blog数据 modelbuilder.entity<blog>().hasqueryfilter(b=>b.isdeleted==1); }
var blgos= bloggingcontext.blogs .fromsql("select * from dbo.blogs") .where(b=>b.blogid>2) .orderbydescending(b=>b.blogid) .select(b=> new{ b.blogid, b.url}) .tolist();
--通过sql server profiler监听的sql,where后面加了isdeleted==1,如下所示: select [b].[blogid], [b].[url] from ( select * from dbo.blogs ) as [b] where ([b].[isdeleted] = 1) and ([b].[blogid] > 2) order by [b].[blogid] desc
//可以在linq查询语句中禁用全局查询筛选器 bloggingcontext.blogs.ignorequeryfilters().tolist();
参考文献:
推荐阅读
-
ASP.NET MVC+EF框架+EasyUI实现权限管理系列(2)-数据库访问层的设计Demo
-
asp.net core 系列 20 EF基于数据模型创建数据库
-
asp.net core系列 29 EF模型配置(查询类型,关系数据库建模)
-
asp.net core系列 35 EF保存数据(2) -- EF系列结束
-
asp.net core系列 32 EF查询数据 必备知识(1)
-
asp.net core 系列 21 EF现有数据库进行反向工程
-
asp.net core系列 31 EF管理数据库架构--必备知识 反向工程
-
asp.net core系列 33 EF查询数据 (2)
-
asp.net core系列 30 EF管理数据库架构--必备知识 迁移
-
ASP.NET MVC+EF框架+EasyUI实现权限管理系列(2)-数据库访问层的设计Demo