sqlser 2005 使用執行計劃來優化你的sql

来源:http://www.cnblogs.com/cq-jiang/archive/2017/10/22/7711680.html
-Advertisement-
Play Games

一:sqlserver 執行計劃介紹 sqlserver 執行計是在sqlser manager studio 工具中打開,是檢查一條sql執行效率的工具。建議配合SET STATISTICS IO ON等語句來一起使用,執行計劃是從右向左看,耗時高的一般顯示在右邊,我們知道,sqlserver 查 ...


一:sqlserver 執行計劃介紹

     sqlserver 執行計是在sqlser manager studio 工具中打開,是檢查一條sql執行效率的工具。建議配合SET STATISTICS IO ON等語句來一起使用,執行計劃是從右向左看,耗時高的一般顯示在右邊,我們知道,sqlserver 查詢資料庫的方式為:

  1:表掃描(table scan) 查詢速度最慢.

  2:聚集索引掃描(Clustered Index Scan),按聚集索引逐行進行查詢,效率比表掃描高,但速度還是慢.

  3:索引掃描(index scan)效率比聚集索引快,根據索引濾出部分數據在進行逐行檢查。

  4;索引檢查(index seek) 效率比索引掃描還要快,根據索引定位記錄所在位置再取出記錄.

  5:聚集索引掃描(Clustered Index Seek) 效率最快,直接根據聚集索引獲取記錄。

當發現某個查詢比較慢時,可以首先檢查哪些操作的成本比較高,再看看那些操作在查找記錄時, 是不是【Table Scan】或者【Clustered Index Scan】,如果確實和這二種操作類型有關,則要考慮增加索引來解決了,sqlser 索引有兩種,聚集索引和非聚集索引,聚集索引是一張表只能有一個,比如id,非聚集索引可以有多個,聚集索引是順序排列的類似於字典查找拼音a、b、c……和字典文字內容順序是相同的,非聚集索引與內容是非順序排列的,類似字典偏旁查找時,同一個偏旁‘王’的漢字可能一個在第1頁一個在第5頁。

二:創建測試表

create table shopping_user(uId bigint primary key,uName varchar(10));
create table shopping_goods_category(cId bigint primary key,cName varchar(20));
create table shopping_goods(gId bigint primary key,gName varchar(50),gcId bigint,gPrice int);
create table shopping_order(oId bigint primary key,oUserId bigint,oAddTime datetime,oGoodsId bigint,oMoney int);
  

  創建測試sql

declare @index int;
set @index = 1;
while(@index<=10)
begin
    insert into shopping_user (uId,uName) values(@index,'user'+cast(@index as varchar(10)));
    set @index = @index+1;
end;

insert into shopping_goods_category (cid,cName) values(1,'水果');
insert into shopping_goods_category (cid,cName) values( 2,'電腦');
insert into shopping_goods_category (cid,cName) values (3,'手機');
insert into shopping_goods_category (cid,cName) values (4,'服裝');
insert into shopping_goods_category (cid,cName) values (5,'食品');

------ 商品表sql

declare @index int;
declare @num int;
set @index = 1;
set @num = 10000;
begin
    while(@index<=100*@num)
    begin
        if @index<=10*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,1,'水果'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end;
        else if @index >10*@num and @index <=20*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,1,'水果'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end;
        else if @index >20*@num and @index <=30*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,2,'電腦'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end;
        else if @index >30*@num and @index <=40*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,2,'電腦'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end;
        else if @index >40*@num and @index <=50*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,3,'手機'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end;
        else if @index >50*@num and @index <=60*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,3,'手機'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end; 
        else if @index >60*@num and @index <=70*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,4,'服裝'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end; 
        else if @index >70*@num and @index <=80*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,4,'服裝'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end; 
        else if @index >80*@num and @index <=90*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,5,'食品'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end; 
        else if @index >90*@num and @index <=100*@num
            begin
                insert into shopping_goods (gId,gcId,gName,gPrice)
                values (@index,5,'食品'+cast (@index as varchar(10)),cast( floor(rand()*100) as int) );
            end; 
        set @index = @index+1;
    end; 
end;


------- 訂單表sql

declare @index int;
declare @num int;
declare @timeNum int;
declare @userId int;
declare @goodsId int; 
declare @money int;
declare @addTime varchar(30);
set @index = 1;
set @num = 10000; 
set @timeNum = 0;
set @userId = 1;
set @goodsid = 1;
set @money = 100;
set @addTime = '';
begin
    while(@index<=100*@num)
    begin
        set @timeNum = cast( floor(rand()*30)+1 as int)
    set @userId = cast( floor(rand()*99)+1 as int)
    set @money = cast ( floor(rand()*5000)+@userId as int)
    set @addTime = dateadd(day,@timeNum,getdate())
    set @goodsId = cast( floor(rand()*999999)+1 as int)
        if @index<=10*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end;
        else if @index >10*@num and @index <=20*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end;
        else if @index >20*@num and @index <=30*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end;
        else if @index >30*@num and @index <=40*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end;
        else if @index >40*@num and @index <=50*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end;
        else if @index >50*@num and @index <=60*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end; 
        else if @index >60*@num and @index <=70*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end; 
        else if @index >70*@num and @index <=80*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end; 
        else if @index >80*@num and @index <=90*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end; 
        else if @index >90*@num and @index <=100*@num
            begin
                insert into shopping_order (oid,oUserId,oAddTime,oGoodsId,oMoney)
                values (@index,@userId,@addTime,@goodsId,@money );
            end; 
    
    set @index = @index+1;
    end;
    
end;

  創建索引

create index gcid_index on shopping_goods (gcid);
create index userid_index on shopping_order(ouserid);
create index goodsid_index on shopping_order(ogoodsid);

三:執行計劃分析

  這裡使用上一篇文章sql語句百萬數據量優化方案中提到的,in和exists來分析,sql語句如下:

SET STATISTICS IO ON

select top 20 * from shopping_order where exists (
select top 10 gid from shopping_goods where gcid =2 and ogoodsid = gid order by gprice desc)

select top 20 * from shopping_order where goodsid in (
select top 10 gid from shopping_goods where gcid =2 order by gprice desc)
 

-- DBCC DROPCLEANBUFFERS 

  

從上圖中發現,使用exists,開銷最大的是,使用聚集索引查找,而使用in,第一次操作(從右各左看),就使用了聚集索引掃描,in的效果明顯差。我們再來看聚集索引查找結果,聚集索引返回的行數是20,見下圖.

 

然後我們來看使用in查詢,聚集索引掃描,查詢結果卻是20w

 

接著我們來看使用in查詢,第二個開銷大的排序,從剛纔查詢出來的20w數據中,order by desc 返回前20條數據。

此處我們還可以使用SET STATISTICS IO ON來查詢這兩者的io開銷: 

    掃描計數:執行的掃描次數;

    邏輯讀取:從數據緩存讀取的頁數;

  物理讀取:從磁碟讀取的頁數;

    預讀:為進行查詢而放入緩存的頁數

重要:如果對於一個SQL查詢有多種寫法,那麼這四個值中的邏輯讀(logical reads)決定了哪個是最優化的。

 

從上圖中發現,exists查詢:shopping_order表掃描次數是2,邏輯讀取是80,shopping_goods表,掃描次數是1,邏輯讀取是6次,

          而in  shopping_order表掃描次數是2,邏輯讀取是55,shopping_goods表,掃描次數是5,邏輯讀取是5247次,當然工作中的sql肯定要複雜得多,但我們可以藉助這個工具來找到需要優化的sql,當然這也只是執行計劃,可能實際執行的效率和這個計劃有出入,但我們還是可以借鑒執行計劃來找到其中的不足。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 介紹 責任鏈模式(Chain of responsibility)是使多個對象都有機會處理請求,從而避免請求的發送者和接受者之間的耦合關係。將對象連成一條鏈,並沿著這條鏈傳遞該請求,直到有一個對象處理他為止。 請求以後,從第一個對象開始,鏈中收到請求的對象要麼親自處理它,要麼轉發給鏈中的下一個候選者 ...
  • 久違的一文了,有一個多月沒有更新我這個博文了,太懶了;今天突然想起來就把這一文寫了。上回簡單說了下ES6的語法,這回我就做了一個sass的綜合項目練習,我用sass寫了一個頁面。只是一個頁面佈局的練習,沒怎麼運用到js。先貼個代碼先: 這個app.js的代碼。 這個 function.scss,在s ...
  • 1. 前言 最近學習ABP架構 搭建ASP.NET Core with Angular遇到了些問題,折騰了一個禮拜最終在今天解決了,想想這個過程的痛苦就想利用博客記錄下來。其實一直想寫博客,但因為 時間忙和惰性一直沒有寫。之前也遇到些問題,費了很大勁最後解決了,但過了好久又遇到類似問題,但想不起來之 ...
  • 一, onmouseover 和 onmouseout 事件 onmouseover 和 onmouseout 事件可用於在用戶的滑鼠移至 HTML 元素上方或移出元素時觸發函數。 一個小例:滑鼠未在上面前 移到上面後 二, onmousedown、onmouseup 以及 onclick 事件 o ...
  • 前言 之前翻博客園的時候,看到有人朋友分享阿裡巴巴的面試題,其中有一道題就是關於ES6轉ES5 原理的,當時我看到感覺到自己離去阿裡巴巴的路還很遠啊,像我們大部分做開發的時候,都只知其然不知 其所以然,本著好學的態度我也去網上搜了下關於Babel es6轉es5的原理,但是很多都是枯燥難懂,整理了一 ...
  • 首先#pragma在本質上是聲明,常用的功能就是註釋,尤其是給Code分段註釋;而且它還有另一個強大的功能是處理編譯器警告,但卻沒有上一個功能用的那麼多。 clang diagnostic 是#pragma 第一個常用命令: 常見的用法 1. 方法棄用告警 2. 不相容指針類型 3. 迴圈引用 4. ...
  • 服務是Android四大組件之一,與Activity一樣,代表可執行程式。但Service不像Activity有可操作的用戶界面,它是一直在後臺運行。用通俗易懂點的話來說: 如果某個應用要在運行時向用戶呈現可操作的信息就應該選擇Activity,如果不是就選擇Service。 Service的生命周 ...
  • 在按照第三方sdk文檔中的Emedded Binaries 中加入了他們的framework,在刪除這下麵的對應的framework後,問題就得到瞭解決 發下有個英文的頁面也是涉及這個問題的, 描述的很詳細. 鏈接: https://stackoverflow.com/questions/41786 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...