99999久久久久久亚洲,欧美人与禽猛交狂配,高清日韩av在线影院,一个人在线高清免费观看,啦啦啦在线视频免费观看www

熱線(xiàn)電話(huà):13121318867

登錄
首頁(yè)精彩閱讀SQL Server中存儲(chǔ)過(guò)程比直接運(yùn)行SQL語(yǔ)句慢的原因
SQL Server中存儲(chǔ)過(guò)程比直接運(yùn)行SQL語(yǔ)句慢的原因
2016-12-13
收藏

SQL Server中存儲(chǔ)過(guò)程比直接運(yùn)行SQL語(yǔ)句慢的原因

在很多的資料中都描述說(shuō)SQLSERVER的存儲(chǔ)過(guò)程較普通的SQL語(yǔ)句有以下優(yōu)點(diǎn):

存儲(chǔ)過(guò)程只在創(chuàng)造時(shí)進(jìn)行編譯即可,以后每次執(zhí)行存儲(chǔ)過(guò)程都不需再重新編譯,而我們通常使用的SQL語(yǔ)句每執(zhí)行一次就編譯一次,所以使用存儲(chǔ)過(guò)程可提高數(shù)據(jù)庫(kù)執(zhí)行速度。

經(jīng)常會(huì)遇到復(fù)雜的業(yè)務(wù)邏輯和對(duì)數(shù)據(jù)庫(kù)的操作,這個(gè)時(shí)候就會(huì)用SP來(lái)封裝數(shù)據(jù)庫(kù)操作。當(dāng)對(duì)數(shù)據(jù)庫(kù)進(jìn)行復(fù)雜操作時(shí)(如對(duì)多個(gè)表進(jìn)行 Update,Insert,Query,Delete時(shí)),可將此復(fù)雜操作用存儲(chǔ)過(guò)程封裝起來(lái)與數(shù)據(jù)庫(kù)提供的事務(wù)處理結(jié)合一起使用??梢詷O大的提高數(shù)據(jù) 庫(kù)的使用效率,減少程序的執(zhí)行時(shí)間,這一點(diǎn)在較大數(shù)據(jù)量的數(shù)據(jù)庫(kù)的操作中是非常重要的。在代碼上看,SQL語(yǔ)句和程序代碼語(yǔ)句的分離,可以提高程序代碼的 可讀性。

存儲(chǔ)過(guò)程可以設(shè)置參數(shù),可以根據(jù)傳入?yún)?shù)的不同重復(fù)使用同一個(gè)存儲(chǔ)過(guò)程,從而高效的提高代碼的優(yōu)化率和可讀性。

安全性高,可設(shè)定只有某此用戶(hù)才具有對(duì)指定存儲(chǔ)過(guò)程的使用權(quán)存儲(chǔ)過(guò)程的種類(lèi):

系統(tǒng)存儲(chǔ)過(guò)程:以sp_開(kāi)頭,用來(lái)進(jìn)行系統(tǒng)的各項(xiàng)設(shè)定.取得信息.相關(guān)管理工作,如 sp_help就是取得指定對(duì)象的相關(guān)信息。

擴(kuò)展存儲(chǔ)過(guò)程 以XP_開(kāi)頭,用來(lái)調(diào)用操作系統(tǒng)提供的功能

exec master..xp_cmdshell ‘ping 10.8.16.1’

用戶(hù)自定義的存儲(chǔ)過(guò)程,這是我們所指的存儲(chǔ)過(guò)程常用格式

模版:Create procedure procedue_name [@parameter data_type][output]
[with]{recompile|encryption} as sql_statement

解釋?zhuān)簅utput:表示此參數(shù)是可傳回的

with {recompile|encryption} recompile:表示每次執(zhí)行此存儲(chǔ)過(guò)程時(shí)都重新編譯一次;encryption:所創(chuàng)建的存儲(chǔ)過(guò)程的內(nèi)容會(huì)被加密。

但是最近我們項(xiàng)目組中有人寫(xiě)了一個(gè)存儲(chǔ)過(guò)程,其計(jì)算時(shí)間為1個(gè)小時(shí)47分鐘,而有的時(shí)候運(yùn)行時(shí)間都超過(guò)了兩個(gè)小時(shí),同事描述說(shuō)如果將存儲(chǔ)過(guò)程中的語(yǔ)句拿出來(lái)直接運(yùn)行也就10分鐘左右就運(yùn)行完畢,我沒(méi)當(dāng)回事,但是今天我自己寫(xiě)的存儲(chǔ)過(guò)程也遇到了這個(gè)問(wèn)題,在查找資料后原因終于找到了原因,原來(lái)是Parameter sniffing問(wèn)題。

下面看我是如何將運(yùn)行一個(gè)小時(shí)以上的存儲(chǔ)過(guò)程優(yōu)化成在一分鐘之內(nèi)完成的:

原存儲(chǔ)過(guò)程

CREATE PROCEDURE [dbo].[pro_ImAnalysis_daily]

@THEDATE VARCHAR(30)

AS

BEGIN

IF @THEDATE IS NULL

BEGIN

SET @THEDATE=CONVERT(VARCHAR(30),GETDATE()-1,112);

END


DELETE FROM RPT_IM_USERINFO_DAILY WHERE THEDATE=@THEDATE;


INSERT RPT_IM_USERINFO_DAILY (THEDATE,ALLUSER,NEWUSER)

SELECT AA.THEDATE,ALLUSER,NEWUSER

FROM

( ( SELECT THEDATE,COUNT(DISTINCT USERID) ALLUSER

FROM FACT

WHERE THEDATE=@THEDATE

GROUP BY THEDATE

) AA

LEFT JOIN

(SELECT THEDATE,COUNT(DISTINCT USERID) NEWUSER

FROM FACT T1

WHERE NOT EXISTS(

SELECT 1

FROM FACT T2

WHERE T2.THEDATE<@THEDATE

AND T1.USERID=T2.USERID)

AND T1.THEDATE=@THEDATE

GROUP BY THEDATE

) BB

ON AA.THEDATE=BB.THEDATE);

GO

每日?qǐng)?zhí)行:exec pro_ImAnalysis_daily @thedate=null
耗時(shí):1小時(shí)47分~2小時(shí)13分

經(jīng)過(guò)查找資料,原因如下(由于源文是一篇英文,有些地方寫(xiě)的我不是特別清楚,原文見(jiàn)http://groups.google.com/group/microsoft.public.sqlserver.server/msg/ad37d8aec76e2b8f?hl=en&lr=&ie=UTF-8&oe=UTF-8):

SQL Server中有一個(gè)叫做 “Parameter sniffing”的特性。SQL Server在存儲(chǔ)過(guò)程執(zhí)行之前都會(huì)制定一個(gè)執(zhí)行計(jì)劃。在上面的例子中,SQL在編譯的時(shí)候并不知道@thedate的值是多少,所以它在執(zhí)行執(zhí)行計(jì)劃的時(shí)候就要進(jìn)行大量的猜測(cè)。假設(shè)傳遞給@thedate的參數(shù)大部分都是非空字符串,而FACT表中有40%的thedate字段都是null,那么SQL Server就會(huì)選擇全表掃描而不是索引掃描來(lái)對(duì)參數(shù)@thedate制定執(zhí)行計(jì)劃。全表掃描是在參數(shù)為空或?yàn)?的時(shí)候最好的執(zhí)行計(jì)劃。但是全表掃描嚴(yán)重影響了性能。

假設(shè)你第一次使用了Exec pro_ImAnalysis_daily @thedate=’20080312’那么SQL Server就會(huì)使用20080312這個(gè)值作為下次參數(shù)@thedate的執(zhí)行計(jì)劃的參考值,而不會(huì)進(jìn)行全表掃描了,但是如果使用@thedate=null,則下次執(zhí)行計(jì)劃就要根據(jù)全表掃描進(jìn)行了。

有兩種方式能夠避免出現(xiàn)“Parameter sniffing”問(wèn)題:

(1)通過(guò)使用declare聲明的變量來(lái)代替參數(shù):使用set @variable=@thedate的方式,將出現(xiàn)@thedate的sql語(yǔ)句全部用@variable來(lái)代替。

(2)  將受影響的sql語(yǔ)句隱藏起來(lái),比如:

a) 將受影響的sql語(yǔ)句放到某個(gè)子存儲(chǔ)過(guò)程中,比如我們?cè)贎thedate設(shè)置成為今天后再調(diào)用一個(gè)字存儲(chǔ)過(guò)程將@thedate作為參數(shù)傳入就可以了。數(shù)據(jù)分析培訓(xùn)

b) 使用sp_executesql來(lái)執(zhí)行受影響的sql。執(zhí)行計(jì)劃不會(huì)被執(zhí)行,除非sp_executesql語(yǔ)句執(zhí)行完。

c) 使用動(dòng)態(tài)sql(”EXEC(@sql)”來(lái)執(zhí)行受影響的sql。

采用(1)的方法改造例子中的存儲(chǔ)過(guò)程,如下:

ALTER PROCEDURE [dbo].[pro_ImAnalysis_daily]

@var_thedate VARCHAR(30)

AS

BEGIN

declare @THEDATE VARCHAR(30)

IF @var_thedate IS NULL

BEGIN

SET @var_thedate=CONVERT(VARCHAR(30),GETDATE()-1,112);

END


SET @THEDATE=@var_thedate;

DELETE FROM RPT_IM_USERINFO_DAILY WHERE THEDATE=@THEDATE;


INSERT RPT_IM_USERINFO_DAILY (THEDATE,ALLUSER,NEWUSER)

SELECT AA.THEDATE,ALLUSER,NEWUSER

FROM

( ( SELECT THEDATE,COUNT(DISTINCT USERID) ALLUSER

FROM FACT

WHERE THEDATE=@THEDATE

GROUP BY THEDATE

) AA

LEFT JOIN

(SELECT THEDATE,COUNT(DISTINCT USERID) NEWUSER

FROM FACT T1

WHERE NOT EXISTS(

SELECT 1

FROM FACT T2

WHERE T2.THEDATE<@THEDATE

AND T1.USERID=T2.USERID)

AND T1.THEDATE=@THEDATE

GROUP BY THEDATE

) BB

ON AA.THEDATE=BB.THEDATE);

GO


測(cè)試執(zhí)行速度為10分鐘,我又檢查了一下這個(gè)SQL,發(fā)現(xiàn)這個(gè)SQL有問(wèn)題,這個(gè)SQL使用了not exists,在一個(gè)大表里面使用not exists是不太明智的,所以,我又對(duì)這個(gè)sql進(jìn)行了改進(jìn),改成如下:

ALTER PROCEDURE [dbo].[pro_ImAnalysis_daily]

@var_thedate VARCHAR(30)


AS

BEGIN

declare @THEDATE VARCHAR(30)

IF @var_thedate IS NULL

BEGIN

SET @var_thedate=CONVERT(VARCHAR(30),GETDATE()-1,112);

END


SET @THEDATE=@var_thedate;

DELETE FROM RPT_IM_USERINFO_DAILY WHERE THEDATE=@THEDATE;


INSERT RPT_IM_USERINFO_DAILY(THEDATE,ALLUSER,NEWUSER)

select @thedate as thedate,

count(distinct case when today>0 then userid else null end) as alluser,

count(distinct case when dates=0 then userid else null end) as newuser

from

(

select userid,

count(CASE WHEN thedate>=@thedate then null else thedate end) as dates,

count(case when thedate=@thedate then thedate else null end) as today

from   FACT

group by userid

)as fact

GO

測(cè)試結(jié)果為30ms以下。

數(shù)據(jù)分析咨詢(xún)請(qǐng)掃描二維碼

若不方便掃碼,搜微信號(hào):CDAshujufenxi

數(shù)據(jù)分析師資訊
更多

OK
客服在線(xiàn)
立即咨詢(xún)
客服在線(xiàn)
立即咨詢(xún)
') } function initGt() { var handler = function (captchaObj) { captchaObj.appendTo('#captcha'); captchaObj.onReady(function () { $("#wait").hide(); }).onSuccess(function(){ $('.getcheckcode').removeClass('dis'); $('.getcheckcode').trigger('click'); }); window.captchaObj = captchaObj; }; $('#captcha').show(); $.ajax({ url: "/login/gtstart?t=" + (new Date()).getTime(), // 加隨機(jī)數(shù)防止緩存 type: "get", dataType: "json", success: function (data) { $('#text').hide(); $('#wait').show(); // 調(diào)用 initGeetest 進(jìn)行初始化 // 參數(shù)1:配置參數(shù) // 參數(shù)2:回調(diào),回調(diào)的第一個(gè)參數(shù)驗(yàn)證碼對(duì)象,之后可以使用它調(diào)用相應(yīng)的接口 initGeetest({ // 以下 4 個(gè)配置參數(shù)為必須,不能缺少 gt: data.gt, challenge: data.challenge, offline: !data.success, // 表示用戶(hù)后臺(tái)檢測(cè)極驗(yàn)服務(wù)器是否宕機(jī) new_captcha: data.new_captcha, // 用于宕機(jī)時(shí)表示是新驗(yàn)證碼的宕機(jī) product: "float", // 產(chǎn)品形式,包括:float,popup width: "280px", https: true // 更多配置參數(shù)說(shuō)明請(qǐng)參見(jiàn):http://docs.geetest.com/install/client/web-front/ }, handler); } }); } function codeCutdown() { if(_wait == 0){ //倒計(jì)時(shí)完成 $(".getcheckcode").removeClass('dis').html("重新獲取"); }else{ $(".getcheckcode").addClass('dis').html("重新獲取("+_wait+"s)"); _wait--; setTimeout(function () { codeCutdown(); },1000); } } function inputValidate(ele,telInput) { var oInput = ele; var inputVal = oInput.val(); var oType = ele.attr('data-type'); var oEtag = $('#etag').val(); var oErr = oInput.closest('.form_box').next('.err_txt'); var empTxt = '請(qǐng)輸入'+oInput.attr('placeholder')+'!'; var errTxt = '請(qǐng)輸入正確的'+oInput.attr('placeholder')+'!'; var pattern; if(inputVal==""){ if(!telInput){ errFun(oErr,empTxt); } return false; }else { switch (oType){ case 'login_mobile': pattern = /^1[3456789]\d{9}$/; if(inputVal.length==11) { $.ajax({ url: '/login/checkmobile', type: "post", dataType: "json", data: { mobile: inputVal, etag: oEtag, page_ur: window.location.href, page_referer: document.referrer }, success: function (data) { } }); } break; case 'login_yzm': pattern = /^\d{6}$/; break; } if(oType=='login_mobile'){ } if(!!validateFun(pattern,inputVal)){ errFun(oErr,'') if(telInput){ $('.getcheckcode').removeClass('dis'); } }else { if(!telInput) { errFun(oErr, errTxt); }else { $('.getcheckcode').addClass('dis'); } return false; } } return true; } function errFun(obj,msg) { obj.html(msg); if(msg==''){ $('.login_submit').removeClass('dis'); }else { $('.login_submit').addClass('dis'); } } function validateFun(pat,val) { return pat.test(val); }