首页 > 数据库 >SqlServer SQL语句或存储过程运行慢 使用 WITH RECOMP ILE 或 OPTION (RECOMPILE)(重新编译)

SqlServer SQL语句或存储过程运行慢 使用 WITH RECOMP ILE 或 OPTION (RECOMPILE)(重新编译)

时间:2024-07-17 10:27:00浏览次数:21  
标签:ILE OPTION RECOMPILE 查询 编译 SQL 执行

如果您的存储过程包含参数可以重新申明变量把参数接收下,可能解决你过程执行慢的原因。如果未能解决,请参考以下文章内容:

WITH RECOMPILE 子句可以在以下地方使用:

一种是当你创建一个过程时,例如:

CREATE PROCEDURE MySP WITH RECOMPILE AS
这指示 SQL Server 在每次调用时重新编译整个过程,并且从不缓存该计划。

另一种情况是当你调用一个过程时:

EXEC MySP WITH RECOMPILE
这告诉 SQL Server 重新编译此调用的过程,绕过缓存中的内容。我不知道新计划是否存储在缓存中,或者是否被丢弃。

这两个都是至少从 SQL Server 4.x 开始就存在于产品中的旧功能。

从SQL 2005开始,有一个更好的选择,那就是语句级提示OPTION(RECOMPILE):

SELECT ...
FROM   tbl
WHERE Id = @id OR @id NULL
OPTION (RECOMPILE)
        这告诉 SQL Server 每次执行此语句时都重新编译它。这比 WITH RECOMPILE 更强大,原因有二。一是如果这是过程中唯一一个每次都需要重新编译的语句,则通过不重新编译其余部分可以节省资源。但真正强大的是 @id 可以作为常量处理,因此当您有 @id 时可以进行索引查找,而当您没有 @id 时可以进行表扫描。使用 WITH RECOMPILE 无法获得这种效果,因为编译发生在过程启动时,并且 @id 的运行时值未知。

        因此,WITH RECOMPILE 和 CREATE PROCEDURE 实际上是一个已停用的功能,您很少(如果有的话)有理由使用它。我认为 EXEC WITH RECOMPILE 从未被广泛使用过。

选项(重新编译):
首先,让我们创建一个包含关键字OPTION(RECOMPILE)的存储过程。

CREATE OR ALTER PROC GetCustomerOrders (@CustomerID INT)
AS
SELECT *
FROM WideWorldImporters.Sales.Orders
WHERE CustomerID = @CustomerID
OPTION (RECOMPILE)

现在在 SQL Server Management Studio (SSMS) 中为查询窗口启用执行计划。

接下来,让我们使用两个不同的参数运行以下两个存储过程。

EXEC GetCustomerOrders 1060
EXEC GetCustomerOrders 90
GO

        当我们看到查询的执行计划时,您会注意到,这次执行计划对行有一个正确的估计,因此查询正在消耗优化特定执行计划所需的资源。

        当查询需要更多资源时,它会占用更多资源,当需要更少资源时,它会使用适当的资源。这是一个很大的调整,SQL Server 每次执行存储过程时都会进行调整。 


参考:SQL SERVER - Parameter Sniffing and OPTION (RECOMPILE) - SQL Authority with Pinal Dave

Queries with OPTION (RECOMPILE) and Query Store - Erin Stellato

使用 Transact-SQLUsing Transact-SQL:

1、连接到数据库引擎。

2、从标准栏中选择“新建查询”。

3、将以下示例复制并粘贴到查询窗口中,然后选择“执行”。此示例创建过程定义。

USE AdventureWorks2022;  
GO  
IF OBJECT_ID ( 'dbo.uspProductByVendor', 'P' ) IS NOT NULL   
    DROP PROCEDURE dbo.uspProductByVendor;  
GO  
CREATE PROCEDURE dbo.uspProductByVendor @Name varchar(30) = '%'  
WITH RECOMPILE  
AS  
    SET NOCOUNT ON;  
    SELECT v.Name AS 'Vendor name', p.Name AS 'Product name'  
    FROM Purchasing.Vendor AS v   
    JOIN Purchasing.ProductVendor AS pv   
      ON v.BusinessEntityID = pv.BusinessEntityID   
    JOIN Production.Product AS p   
      ON pv.ProductID = p.ProductID  
    WHERE v.Name LIKE @Name;  
    
使用 WITH RECOMPILE 选项重新编译存储过程
选择“新建查询”,然后将以下代码示例复制并粘贴到查询窗口中,然后选择“执行”。这将执行该过程并重新编译该过程的查询计划。

USE AdventureWorks2022;  
GO  
EXECUTE HumanResources.uspProductByVendor WITH RECOMPILE;  
GO

使用 sp_recompile 重新编译存储过程
选择“新建查询”,然后将以下示例复制并粘贴到查询窗口中,然后选择“执行”。这不会执行该过程,但它会标记要重新编译的过程,以便在下次执行该过程时更新其查询计划。

USE AdventureWorks2022;  
GO  
EXEC sp_recompile N'dbo.uspProductByVendor';   
GO 

参考:Recompile a Stored Procedure - SQL Server | Microsoft Learn 

OPTION(RECOMPILE)总是更快;为什么?

        有时使用OPTION(RECOMPILE)是有意义的。根据我的经验,唯一可行的选择是使用动态 SQL。在探索这是否适合您的情况之前,我建议您重建统计信息。这可以通过运行以下命令来完成:

EXEC sp_updatestats
然后重新创建执行计划。这将确保在创建执行计划时将使用最新信息。

OPTION(RECOMPILE)每次执行查询时都会重新构建执行计划。我从未听说过这样的描述,但creates a new lookup strategy也许我们只是对同一件事使用了不同的术语。

        创建存储过程时(我怀疑您正在从 .NET 调用临时 SQL,但如果您使用参数化查询,那么这最终将成为存储过程调用)SQL Server 尝试根据数据库中的数据和传入的参数确定此查询的最有效执行计划(参数嗅探),然后缓存该计划。这意味着,如果您在数据库中有 10 条记录的情况下创建查询,然后在有 100,000,000 条记录时执行该查询,则缓存的执行计划可能不再是最有效的。

        总结一下 - 我看不出有任何理由OPTION(RECOMPILE)会在这里带来好处。我怀疑你只需要更新你的统计数据和执行计划。重建统计数据可能是 DBA 工作的一个重要部分,具体取决于你的情况。如果在更新统计数据后仍然遇到问题,我建议发布两个执行计划。

        OPTION (RECOMPILE)为什么更快,我想说,每次执行查询时重新编译执行计划的最佳选择是非常不寻常的。

参考:sql - OPTION (RECOMPILE) is Always Faster; Why? - Stack Overflow 

标签:ILE,OPTION,RECOMPILE,查询,编译,SQL,执行
From: https://blog.csdn.net/hefeng_aspnet/article/details/140129596

相关文章

  • 从零手写实现 nginx-29-try_files 指令
    前言大家好,我是老马。很高兴遇到你。我们为java开发者实现了java版本的nginxhttps://github.com/houbb/nginx4j如果你想知道servlet如何处理的,可以参考我的另一个项目:手写从零实现简易版tomcatminicat手写nginx系列如果你对nginx原理感兴趣,可以阅读:从零......
  • auto,static,const,extern,volatile,register
    auto关键字用于声明变量的生存期为自动,auto修饰的是自动类型的变量,对于局部变量默认就是自动类型的变量,如果没有赋初值它的值就是随机值。static 修饰的变量或者函数有如下特点:static修饰的局部变量,可以延长变量的生命周期(不会被多次初始化)static修饰的全局变量或者函数只......
  • 2024-07-16升级问题:调用自带软件打开文件时 android.os.FileUriExposedException
    2024-07-16升级问题:调用手机自带软件打开文件时,出现以下问题:E/AndroidRuntime:FATALEXCEPTION:mainProcess:rs.tabletcropland,PID:10997android.os.FileUriExposedException:file:///storage/emulated/0/arcgis/%E7%9F%B3%E7%8B%AE%E5%B8%82/Attachment/%E7......
  • Vue3新特性defineOptions和defineModel 面试总结
    在Vue3中,defineOptions和defineModel是两个重要的新特性,它们分别在组件定义和v-model双向绑定方面提供了更为便捷和高效的解决方案。defineOptions定义与用途:defineOptions是Vue3.3+版本中引入的一个宏(macro),用于在<scriptsetup>语法糖中定义组件的选项,如组件名(name)、透传属......
  • MapLibre/Martin | 使用Martin发布MBTiles地图切片包
    什么是MartinMartin是一个高性能的地图切片服务器,使用Rust编写,支持PostGIS,MBTiles,PMTiles。什么是MBTilesMBTiles是个sqlite文件,也就是说MBTiles文件是个单文件数据库。截至本文写作时,最新标准是1.3.MBTIles利用了数据库的索引机制,避免相同内容的切片重复占用空间,同时也有......
  • 报错\corelibrary\think\cache\driver\File.php
    这个错误通常发生在尝试使用 unserialize() 函数反序列化一个存储的缓存数据时,但提供的缓存数据无法被正确解析。在 \core\library\think\cache\driver\File.php 文件的第117行左右,框架尝试从文件中读取缓存数据并反序列化它。以下是一些可能的解决方案:检查缓存文件内容:首......
  • [1033] Run the batch file of the PyQt GUI, but don't show the CMD
    IbuiltaGUIlayoutusingPyQt,andIrunthePythonfilewitha.batfile.However,thismethodinitiallydisplaystheCMDwindowbeforeshowingtheprogram'smainwindow,whichlooksabitodd.TopreventorminimizetheCMDwindowfromappearing,......
  • buying a SIM card and registering a mobile phone number
    Here’sasampleconversationaboutbuyingaSIMcardandregisteringamobilephonenumber:User:Hi,IneedtobuyaSIMcardandregisteramobilephonenumber.Canyouguidemethroughtheprocess?Salesperson:Ofcourse!First,weneedtochooseamo......
  • 木舟0基础学习Java的第十七天(File类使用,IO流)
     File类(路径):文件路径文件夹路径路径:1.绝对路径:固定的路径从盘符开始2.相对路径:相对于某个位置指当前项目下创建功能:Filef=newFile("e:/a.txt");try{booleanb1=f.createNewFile();//新建文件System.out.println(b1);......
  • optional
    作用:用来包装一个可能存在也可能不存在的值当存在时我们可以选择返回其中内容,不存在时返回空;代码示例:#include<iostream>#include<optional>#include<string>#include<fstream>std::optional<std::string>ReadFileAsString(conststd::string&filepath){std::i......