首頁 資料庫 mysql教程 SQLServer : EXEC和sp

SQLServer : EXEC和sp

Jun 07, 2016 pm 03:39 PM
exec sqlserver 使用 摘要

摘要 1,EXEC的使用 2,sp_executesql的使用 MSSQL为我们提供了两种动态执行SQL语句的命令,分别是EXEC和sp_executesql;通常,sp_executesql则更具有优势,它提供了输入输出接口,而EXEC没有。还有一个最大的好处就是利用sp_executesql,能够重用执行计划,这

摘要

1,EXEC的使用

2,sp_executesql的使用

       MSSQL为我们提供了两种动态执行SQL语句的命令,分别是EXEC和sp_executesql;通常,sp_executesql则更具有优势,它提供了输入输出接口,而EXEC没有。还有一个最大的好处就是利用sp_executesql,能够重用执行计划,这就大大提供了执行性能(对于这个我在后面的例子中会详加说明),还可以编写更安全的代码。EXEC在某些情况下会更灵活。除非您有令人信服的理由使用EXEC,否侧尽量使用sp_executesql.

1,EXEC的使用

EXEC命令有两种用法,一种是执行一个存储过程,另一种是执行一个动态的批处理。以下所讲的都是第二种用法。

下面先使用EXEC演示一个例子,代码1

<span>DECLARE</span> @TableName <span>VARCHAR</span>(50),@<span>Sql</span> NVARCHAR(<span>MAX</span>),@OrderID <span>INT</span>;
登入後複製
登入後複製
<span>SET</span> @TableName = <span>'Orders'</span>;
登入後複製
登入後複製
<span>SET</span> @OrderID = 10251;
登入後複製
登入後複製
登入後複製
<span>SET</span> @<span>sql</span> = <span>'SELECT * FROM '</span>+QUOTENAME(@TableName) +<span>'WHERE OrderID = '</span>+<span>CAST</span>(@OrderID <span>AS</span> <span>VARCHAR</span>(10))+<span>' ORDER BY ORDERID DESC'</span>
登入後複製
<span>EXEC</span>(@<span>sql</span>);
登入後複製
登入後複製
登入後複製

注:这里的EXEC括号中只允许包含一个字符串变量,但是可以串联多个变量,如果我们这样写EXEC:

<span>EXEC</span>(<span>'SELECT TOP('</span>+ <span>CAST</span>(@TopCount <span>AS</span> <span>VARCHAR</span>(10)) +<span>')* FROM '</span>+QUOTENAME(@TableName) +<span>' ORDER BY ORDERID DESC'</span>);
登入後複製

 

SQL编译器就会报错,编译不通过,而如果我们这样:

<span>EXEC</span>(@<span>sql</span>+@sql2+@sql3);
登入後複製

编译器就会通过;

 

所以最佳的做法是把代码构造到一个变量中,然后再把该变量作为EXEC命令的输入参数,这样就不会受限制了;

 

EXEC不提供接口

这里的接口是指,它不能执行一个包含一个带变量符的批处理,这里乍一听好像不明白,不要紧,我在下面有一个实例,您一看就知道什么意思.

<span>DECLARE</span> @TableName <span>VARCHAR</span>(50),@<span>Sql</span> NVARCHAR(<span>MAX</span>),@OrderID <span>INT</span>;
登入後複製
登入後複製
<span>SET</span> @TableName = <span>'Orders'</span>;
登入後複製
登入後複製
<span>SET</span> @OrderID = 10251;
登入後複製
登入後複製
登入後複製
<span>SET</span> @<span>sql</span> = <span>'SELECT * FROM '</span>+QUOTENAME(@TableName) +<span>'WHERE <strong>OrderID = @OrderID</strong> ORDER BY ORDERID DESC'</span>
登入後複製
<span>EXEC</span>(@<span>sql</span>);
登入後複製
登入後複製
登入後複製

关键就在SET @sql这一句话中,如果我们运行这个批处理,编译器就会产生一下错误

Msg 137, Level 15, State 2, Line 1
必须声明标量变量 "@OrderID"。

使用EXEC时,如果您想访问变量,必须把变量内容串联到动态构建的代码字符串中,如:<span>SET</span> @<span>sql</span> = <span>'SELECT * FROM '</span>+QUOTENAME(@TableName) +<span>'WHERE OrderID = '</span>+<span>CAST</span>(@OrderID <span>AS</span> <span>VARCHAR</span>(10))+<span>' ORDER BY ORDERID DESC'</span>

串联变量的内容也存在性能方面的弊端。SQL Server为每一个的查询字符串创建新的执行计划,即使查询模式相同也是这样。为演示这一点,先清空缓存中的执行计划

DBCC FREEPROCCACHE (这个不是本文所涉及的内容,您可以查看MS的MSDN)

http://msdn.microsoft.com/zh-cn/library/ms174283.aspx 

将代码1运行3次,分别对@OrderID 赋予下面3个值,10251,10252,10253。然后使用下面的代码查询

<span>SELECT</span> cacheobjtype,objtype,usecounts,<span>sql</span> <span>FROM</span> sys.syscacheobjects <span>WHERE</span> <span>sql</span> <span>NOT</span> <span>LIKE</span> <span>'%cach%'</span> <span>AND</span> <span>sql</span> <span>NOT</span> <span>LIKE</span> <span>'%sys.%'</span> 
登入後複製

点击F5运行,就会出现下面如图所示的查询结果:
SQLServer : EXEC和sp

我们可以看到,每执行一次都要产生一次的编译,执行计划没有得到充分重用。

EXEC除了不支持动态批处理中的输入参数外,他也不支持输出参数。默认情况下,EXEC把查询的输出返回给调用者。例如下面代码返回Orders表中所有的记录数

<span>DECLARE</span> @<span>sql</span> NVARCHAR(<span>MAX</span>)
登入後複製
<span>SET</span> @<span>sql</span> = <span>'SELECT COUNT(ORDERID) FROM Orders'</span>;
登入後複製
登入後複製
<span>EXEC</span>(@<span>sql</span>);
登入後複製
登入後複製
登入後複製
然而,如果你要把输出返回给调用批处理中的变量,事情就没有那么简单了。为此,你必须使用INSERT EXEC语法把输出插入到一个目标表中,然后从这表中获取值后赋给该变量,就像这样:

<span>DECLARE</span> @<span>sql</span> NVARCHAR(<span>MAX</span>),@RecordCount INT
登入後複製
<span>SET</span> @<span>sql</span> = <span>'SELECT COUNT(ORDERID) FROM Orders'</span>;
登入後複製
登入後複製
 
登入後複製
<span>CREATE</span> <span>TABLE</span> #T(TID <span>INT</span>);
登入後複製
INSERT INTO #T <span>EXEC</span>(@<span>sql</span>);
登入後複製
<span>SET</span> @RecordCount = (<span>SELECT</span> TID <span>FROM</span> #T)
登入後複製
<span>SELECT</span> @RecordCount
登入後複製
<span>DROP</span> <span>TABLE</span> #T
登入後複製

2,sp_executesql的使用

sp_executesql命令在SQL Server中引入的比EXEC命令晚一些,它主要为重用执行计划提供更好的支持。

为了和EXEC作一个鲜明的对比,我们看看如果用代码1的代码,把EXEC换成sp_executesql,看看是否得到我们所期望的结果

<span>DECLARE</span> @TableName <span>VARCHAR</span>(50),@<span>sql</span> NVARCHAR(<span>MAX</span>),@OrderID <span>INT</span> ,@sql2 NVARCHAR(<span>MAX</span>);
登入後複製
<span>SET</span> @TableName = <span>'Orders '</span>;
登入後複製
<span>SET</span> @OrderID = 10251;
登入後複製
登入後複製
登入後複製
<span>SET</span> @<span>sql</span> = <span>'SELECT * FROM '</span>+QUOTENAME(@TableName) + <span>' WHERE OrderID = '</span>+<span>CAST</span>(@OrderID <span>AS</span> <span>VARCHAR</span>(50)) + <span>' ORDER BY ORDERID DESC'</span>
登入後複製
<span>EXEC</span> sp_executesql @sql
登入後複製

注意最后一行;

事实证明可以运行;

sp_executesql提供接口

sp_executesql命令比EXEC命令更灵活,因为它提供一个接口,该接口及支持输入参数也支持输出参数。这功能使你可以创建带参数的查询字符串,这样就可以比EXEC更好的重用执行计划,sp_executesql的构成与存储过程非常相似,不同之处在于你是动态构建代码。它的构成包括:代码快,参数声明部分,参数赋值部分。说了这么多,还是看看它的语法吧

EXEC sp_executesql

@stmt = ,--类似存储过程主体

@params = , --类似存储过程参数部分

--类似存储过程调用

@stmt参数是输入的动态批处理,它可以引入输入参数或输出参数,和存储过程的主体语句一样,只不过它是动态的,而存储过程是静态的,不过你也可以在存储过程中使用sp_executesql;

@params参数与定义输入/输出参数的存储过程头类似,实际上和存储过程头的语法完全一样;

@ 与调用存储过程的EXEC部分类似。

为了说明sp_executesql对执行计划的管理优于EXEC,我将使用前面讨论EXEC时用到的代码。

<span>   1:  </span>DECLARE @TableName VARCHAR(50),@sql NVARCHAR(MAX),@OrderID INT;
登入後複製
<span>   2:  </span>SET @TableName = <span>'Orders '</span>;
登入後複製
<span>   3:  </span>SET @OrderID = 10251;
登入後複製
<span>   4:  </span>SET @sql = <span>'SELECT * FROM '</span>+QUOTENAME(@TableName) + <span>' WHERE OrderID = @OID ORDER BY ORDERID DESC'</span>
登入後複製
<span>   5:  </span>EXEC sp_executesql
登入後複製
<span>   6:  </span>    @stmt = @sql,
登入後複製
<span>   7:  </span>    @<span>params</span> = N<span>'@OID AS INT '</span>,
登入後複製
<span>   8:  </span>    @OID = @OrderID
登入後複製

在调用该代码和检查它生成的执行计划前,先清空缓存中的执行计划;

DBCC FREEPROCCACHE

将上面的动态代码执行3次,每次执行都赋予@OrderID 不同的值,然后查询sys.syscacheobjects表,并注意它的输出,优化器只创建了一个备用计划,而且该计划被重用的3次

SELECT cacheobjtype,objtype,usecounts,sql FROM sys.syscacheobjects WHERE sql NOT LIKE <span>'%cache%'</span> AND sql NOT LIKE <span>'%sys.%'</span> AND sql NOT LIKE <span>'%sp_executesql%'</span>
登入後複製

点击F5运行,就会出现如下表所示的结果;

SQLServer : EXEC和sp

sq_executesql的另一个与其接口有关的强大功能是,你可以使用输出参数为调用批处理中的 变量返回值。利用该功能可以避免用临时表返回数据,从而得到更高效的代码和更少的重新编译。定义和使用输出参数的语法与存储过程类似。也就是说,你需要在声明参数时指定OUTPUT子句。例如,下面的静态代码简单的演示了如何从动态批处理中利用输出参数@p把值返回到外部批处理中的变量@i.

DECLARE @sql AS NVARCHAR(12),@i AS INT;
登入後複製
SET @sql = N<span>' SET @p = 10'</span>;
登入後複製
EXEC sp_executesql 
登入後複製
    @stmt = @sql,
登入後複製
    @<span>params</span> = N<span>'@p AS INT OUTPUT'</span>,
登入後複製
    @p = @i OUTPUT
登入後複製
SELECT @i
登入後複製

该代码返回输出10

 

          以上就是EXEC和sp_executesql的主要区别,如果各位看官觉得哪不对或者表达不清楚的,还请多多指出^_^SQLServer : EXEC和sp

 

          作者:兴百放

          时间:2008-11-2 22:30

          网址:Http://xbf321.cnblogs.com

 

 

          参考文献: SQL Server 2005 技术内幕-T-SQL 程序设计

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
4 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1670
14
CakePHP 教程
1428
52
Laravel 教程
1329
25
PHP教程
1273
29
C# 教程
1256
24
sqlserver怎麼匯入mdf文件 sqlserver怎麼匯入mdf文件 Apr 08, 2024 am 11:41 AM

匯入步驟如下:將 MDF 檔案複製到 SQL Server 的資料目錄(通常為 C:\Program Files\Microsoft SQL Server\MSSQL\DATA)。在 SQL Server Management Studio(SSMS)中,開啟資料庫並選擇「附加」。點選“新增”按鈕,選擇 MDF 檔案。確認資料庫名稱,點選確定按鈕即可。

sqlserver資料庫中已存在名為的物件怎麼解決 sqlserver資料庫中已存在名為的物件怎麼解決 Apr 05, 2024 pm 09:42 PM

對於 SQL Server 資料庫中已存在同名對象,需要採取下列步驟:確認物件類型(表格、檢視、預存程序)。如果物件為空,可使用 IF NOT EXISTS 跳過建立。如果物件有數據,使用不同名稱或修改結構。使用 DROP 刪除現有物件(謹慎操作,建議備份)。檢查架構更改,確保沒有引用刪除或重新命名的物件。

crystaldiskmark是什麼軟體? -crystaldiskmark如何使用? crystaldiskmark是什麼軟體? -crystaldiskmark如何使用? Mar 18, 2024 pm 02:58 PM

CrystalDiskMark是一款適用於硬碟的小型HDD基準測試工具,可快速測量順序和隨機讀取/寫入速度。接下來就讓小編為大家介紹一下CrystalDiskMark,以及crystaldiskmark如何使用吧~一、CrystalDiskMark介紹CrystalDiskMark是一款廣泛使用的磁碟效能測試工具,用於評估機械硬碟和固態硬碟(SSD)的讀取和寫入速度和隨機I/O性能。它是一款免費的Windows應用程序,並提供用戶友好的介面和各種測試模式來評估硬碟效能的不同方面,並被廣泛用於硬體評

怎麼查看sqlserver連接埠號 怎麼查看sqlserver連接埠號 Apr 05, 2024 pm 09:57 PM

若要查看 SQL Server 連接埠號碼:開啟 SSMS,連線到伺服器。在物件資源管理器中找到伺服器名稱,右鍵單擊它,然後選擇“屬性”。在「連線」標籤中,查看「TCP 連接埠」欄位。

sqlserver誤刪資料庫怎麼恢復 sqlserver誤刪資料庫怎麼恢復 Apr 05, 2024 pm 10:39 PM

若誤刪 SQL Server 資料庫,可採取下列步驟還原:停止資料庫活動;備份日誌檔案;檢查資料庫日誌;復原選項:從備份還原;從交易日誌還原;使用 DBCC CHECKDB;使用第三方工具。請定期備份資料庫並啟用交易日誌以防止資料遺失。

sqlserver資料庫在哪裡 sqlserver資料庫在哪裡 Apr 05, 2024 pm 08:21 PM

SQL Server 資料庫檔案通常儲存在下列預設位置:Windows: C:\Program Files\Microsoft SQL Server\MSSQL\DATALinux: /var/opt/mssql/data可透過修改資料庫檔案路徑設定來自訂資料庫檔案位置。

BTCC教學:如何在BTCC交易所綁定使用MetaMask錢包? BTCC教學:如何在BTCC交易所綁定使用MetaMask錢包? Apr 26, 2024 am 09:40 AM

MetaMask(中文也叫小狐狸錢包)是一款免費的、廣受好評的加密錢包軟體。目前,BTCC已支援綁定MetaMask錢包,綁定後可使用MetaMask錢包進行快速登錄,儲值、買幣等,且首次綁定還可獲得20USDT體驗金。在BTCCMetaMask錢包教學中,我們將詳細介紹如何註冊和使用MetaMask,以及如何在BTCC綁定並使用小狐狸錢包。 MetaMask錢包是什麼? MetaMask小狐狸錢包擁有超過3,000萬用戶,是當今最受歡迎的加密貨幣錢包之一。它可免費使用,可作為擴充功能安裝在網絡

sqlserver服務無法啟動怎麼辦 sqlserver服務無法啟動怎麼辦 Apr 05, 2024 pm 10:00 PM

當 SQL Server 服務無法啟動時,可採取下列步驟解決:檢查錯誤日誌以確定根本原因。確保服務帳戶具有啟動服務的權限。檢查依賴項服務是否正在執行。禁用防毒軟體。修復 SQL Server 安裝。如果修復不起作用,重新安裝 SQL Server。

See all articles