Entity Framework 4.2 exec sp_executesql 不使用索引(参数嗅探)
我遇到了针对 SQL Server 2008 R2 运行的实体框架 (4.2) 生成的简单 SQL 查询的一些主要性能问题。在某些情况下(但不是全部),EF 使用以下语法:
exec sp_executesql 'DYNAMIC-SQL-QUERY-HERE', @param1...
在其他情况下,只是使用嵌入到查询中的提供的参数来执行原始 SQL。我遇到的问题是,使用 sp_executesql 执行的查询忽略了目标表上的所有索引,导致查询性能极差(通过检查 SSMS 中的执行计划确认)。
经过一番研究,听起来这个问题可能是由“参数嗅探”引起的。如果我像这样附加 OPTION(RECOMPILE) 查询提示:
exec sp_executesql 'DYNAMIC-SQL-QUERY-HERE OPTION(RECOMPILE)', @param1...
将使用目标表上的索引,并且查询执行得非常快。我还尝试在数据库实例上切换用于禁用参数嗅探(4136)的跟踪标志(http://support.microsoft .com/kb/980653),但这似乎没有任何效果。
这给我留下了几个问题:
- 是否有办法将 OPTION(RECOMPILE) 查询提示附加到实体框架生成的 SQL 中?
- 是否有办法阻止实体框架使用 exec sp_executesql,而只是运行原始 SQL?
- 还有其他人遇到这个问题吗?还有其他提示/技巧吗?
其他信息:
- 我确实通过 SSMS 重新启动了数据库实例,但是,我将尝试从服务管理控制台重新启动该服务。
- 参数化设置为 SIMPLE (is_parameterization_forced: 0)
- 针对临时工作负载的优化具有以下设置
- 值:0
- 最小值:0
- 最大值:1
- 使用价值:0
- is_dynamic:1
- is_advanced:1
我还应该提到,如果我在使用以下脚本启用跟踪标志 4136 后通过服务管理控制台重新启动 SQL Server 服务,则会出现实际清除跟踪标志...也许我应该以不同的方式执行此操作...
DBCC TRACEON(4136,-1)
I'm encountering some major performance problems with simple SQL queries generated by the Entity Framework (4.2) running against SQL Server 2008 R2. In some situations (but not all), EF uses the following syntax:
exec sp_executesql 'DYNAMIC-SQL-QUERY-HERE', @param1...
In other situations is simply executes the raw SQL with the provided parameters baked into the query. The problem I'm encountering is that queries executed with the sp_executesql are ignoring all indexes on my target tables, resulting in an extremely poor performing query (confirmed by examining the execution plan in SSMS).
After a bit of research, it sounds like the issue might be caused by 'parameter sniffing'. If I append the OPTION(RECOMPILE) query hint like so:
exec sp_executesql 'DYNAMIC-SQL-QUERY-HERE OPTION(RECOMPILE)', @param1...
The indexes on the target tables are used and the query executes extremely quickly. I've also tried toggling on the trace flag used to disable parameter sniffing (4136) on the database instance (http://support.microsoft.com/kb/980653), however this didn't appear to have any effect whatsoever.
This leaves me with a few questions:
- Is there anyway to append the OPTION(RECOMPILE) query hint to the SQL generated by Entity Framework?
- Is there anyway to prevent Entity Framework from using exec sp_executesql, and instead simply run the raw SQL?
- Is anyone else running into this problem? Any other hints/tips?
Additional Information:
- I did restart the database instance through SSMS, however, I will try restarting the service from the service management console.
- Parameterization is set to SIMPLE (is_parameterization_forced: 0)
- Optimize for adhoc workloads has the following settings
- value: 0
- minimum: 0
- maximum: 1
- value_in_use: 0
- is_dynamic: 1
- is_advanced: 1
I should also mention that if I restart the SQL Server Service via the service management console AFTER enabling trace flag 4136 with the below script, appears to actually clear the trace flag...perhaps I should be doing this a different way...
DBCC TRACEON(4136,-1)
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
tl;dr
更新统计信息
我们有一个带有一个参数(主键)的
delete
查询,通过 EF 调用时大约需要 7 秒才能完成和sp_executesql。手动运行查询,并将参数嵌入到sp_executesql
的第一个参数中,可以使查询快速运行(约 0.2 秒)。添加选项(重新编译)
也有效。当然,由于我们使用的是 EF,所以这两种解决方法对我们来说不可用。可能由于级联外键约束,长时间运行的查询的执行计划非常庞大。当我查看 SSMS 中的执行计划时,我注意到在某些情况下不同步骤之间的箭头比其他步骤更宽,这可能表明 SQL Server 无法做出正确的决策。这让我开始思考统计学。我查看了执行计划中的步骤,看看可疑步骤涉及哪些表。然后我为该表运行了
更新统计表
。然后我重新运行了错误的查询。我又重新运行了一遍。再次确认。它起作用了。我们的表现恢复正常。 (仍然比非 sp_executesql 性能差一些,但是嘿!)事实证明,这只是我们开发环境中的一个问题。 (这是一个大问题,因为它使我们的集成测试花费了很长时间。)在我们的生产环境中,我们运行了一项定期更新所有统计数据的作业。
tl;dr
update statistics
We had a
delete
query with one parameter (the primary key) that took ~7 seconds to complete when called through EF andsp_executesql
. Running the query manually, with the parameter embedded in the first argument tosp_executesql
made the query run quickly (~0.2 seconds). Addingoption (recompile)
also worked. Of course, those two workarounds aren't available to us since were using EF.Probably due to cascading foreign key constraints, the execution plan for the long running query was, uhmm..., huge. When I looked at the execution plan in SSMS I noticed that the arrows between the different steps in some cases were wider than others, possibly indicating that SQL Server had trouble making the right decisions. That led me to thinking about statistics. I looked at the steps in the execution plan to see what table was involved in the suspect steps. Then I ran
update statistics Table
for that table. Then I re-ran the bad query. And I re-ran it again. And again just to make sure. It worked. Our perf was back to normal. (Still somewhat worse than non-sp_executesql
performance, but hey!)It turned out that this was only a problem in our development environment. (And it was a big problem because it made our integration tests take forever.) In our production environment, we had a job running that updated all statistics on a regular basis.
此时,我建议:
将临时工作负载优化设置设置为 true。
如果一段时间后此设置似乎没有帮助,只有那时我才会尝试跟踪标志的额外支持。这些通常被保留作为最后的手段。通过 SQL Server 配置管理器使用命令行设置跟踪标志,而不是在查询窗口中使用全局标志。请参阅http://msdn.microsoft.com/en-us/library/ms187329.aspx
At this point I would recommend:
Set the optimize for ad hoc workloads setting to true.
If after some time this setting doesn't seem to have helped, only then would I try the additional support of the trace flag. These are usually reserved as a last resort. Set the trace flag using the command line via SQL Server Configuration Manager, as opposed to in a query window and using the global flag. See http://msdn.microsoft.com/en-us/library/ms187329.aspx