Heidenreich Link 🚀

SQL Server Query fast but slow from procedure

April 5, 2025

SQL Server Query fast but slow from procedure

Experiencing a irritating show discrepancy wherever your SQL question runs blazingly accelerated connected its ain, however slows to a crawl once known as inside a saved process? You’re not unsocial. This is a communal content that galore SQL Server builders brush. Knowing the causes down this behaviour and figuring out however to troubleshoot it tin importantly contact your exertion’s show and prevention you invaluable clip. This station delves into the communal culprits down this show puzzle and offers actionable options to acquire your saved procedures moving astatine optimum velocity.

Parameter Sniffing

1 of the about predominant causes of this show disparity is parameter sniffing. SQL Server creates an execution program based mostly connected the parameters supplied the archetypal clip a saved process is executed. If consequent calls usage antithetic parameters that would payment from a antithetic execution program, SQL Server whitethorn reuse the first, little businesslike program, ensuing successful slower show. Ideate a question looking for clients successful a circumstantial metropolis. If the archetypal execution searches for clients successful a tiny municipality, the optimized program mightiness affect an scale movement. Nevertheless, if the adjacent call searches for clients successful a ample metropolis, a array scan would apt beryllium much businesslike. Unluckily, SQL Server mightiness implement with the scale movement, starring to show degradation.

Respective options be to code parameter sniffing, together with utilizing the RECOMPILE action, including Action (RECOMPILE) to the question inside the process, oregon utilizing section variables inside the saved process.

Statistic

Outdated oregon inaccurate statistic tin besides lend to dilatory saved process execution. Statistic aid the question optimizer take the about businesslike execution program. If statistic are not ahead-to-day, the optimizer mightiness brand mediocre choices, starring to show points. Commonly updating statistic, particularly last important information adjustments, is important for optimum show. See utilizing the sp_updatestats scheme saved process oregon enabling automated statistic updates.

1 existent-planet illustration active a case whose reporting saved process execution clip ballooned from minutes to hours last a ample information import. Updating the statistic resolved the content, dramatically bettering show.

Program Caching

Piece program caching is mostly generous, it tin typically origin issues akin to parameter sniffing. Equal if statistic are ahead-to-day, a cached program mightiness go suboptimal owed to information adjustments. Clearing the program cache, both for a circumstantial saved process oregon for the full server, tin generally resoluteness show points. Nevertheless, usage this with warning, arsenic it tin negatively contact general server show if utilized excessively. See utilizing DBCC FREEPROCCACHE for focused program clearing.

Present’s a simplified illustration of however program caching tin origin points. Ideate a saved process that joins 2 tables. Initially, 1 array is overmuch smaller than the another, truthful a nested loop articulation is the optimum scheme. Complete clip, the smaller array grows importantly. Present, a hash articulation would beryllium much businesslike. Nevertheless, if the first program is cached, SQL Server mightiness proceed utilizing the little businesslike nested loop articulation.

Indexing

Lacking oregon inefficient indexes are different communal origin of dilatory queries, some inside and extracurricular saved procedures. Guarantee your tables person due indexes for the queries being executed inside the saved process. Analyse the question execution program to place lacking oregon underutilized indexes. Instruments similar SQL Server Profiler and Database Motor Tuning Advisor tin aid place possible indexing enhancements.

Cardinal areas to analyze once optimizing indexes see the columns utilized successful Wherever clauses, Articulation circumstances, and Command BY clauses. See creating masking indexes to additional optimize show.

Figuring out Show Bottlenecks

Instruments similar SQL Server Profiler and the execution program itself supply invaluable insights into the show bottlenecks inside your saved procedures. By analyzing the execution program, you tin place agelong-moving operations, extreme I/O, oregon inefficient joins. These instruments tin aid pinpoint the circumstantial areas of the saved process that necessitate optimization.

  • Usage SQL Server Profiler to seizure show information.
  • Analyse the execution program to pinpoint bottlenecks.
  1. Place the dilatory saved process.
  2. Execute the process successful SQL Server Direction Workplace.
  3. Show the Estimated Execution Program.
  4. Analyse the program for costly operations.

Arsenic an adept quoted successful Brent Ozar’s weblog, “Indexing is similar giving your database a roadmap. With out it, the question optimizer is driving unsighted.”

Another Elements and Champion Practices

Respective another components tin lend to dilatory saved process show. These see web latency, extreme blocking, and inefficient codification inside the saved process itself. Reviewing the codification for redundant operations oregon inefficient algorithms tin generally output important show features. Moreover, guarantee your database server has adequate sources, together with CPU, representation, and disk I/O.

Featured Snippet: To rapidly place show points, analyse the question execution program inside SQL Server Direction Workplace. This ocular cooperation pinpoints bottlenecks similar inefficient joins oregon lacking indexes.

Larn much astir precocious SQL Server methods. [Infographic Placeholder: Illustrating the contact of parameter sniffing connected question show]

  • Commonly replace statistic.
  • See utilizing the WITH RECOMPILE action judiciously.

By addressing these possible issues, you tin importantly better the show of your SQL Server saved procedures and guarantee your functions tally easily. Retrieve to totally trial immoderate modifications earlier deploying them to exhibition. It’s frequently a operation of elements that pb to dilatory execution, requiring a methodical attack to troubleshooting and optimization. Studying to efficaciously usage the disposable diagnostic instruments and knowing the intricacies of SQL Server’s question optimizer volition empower you to sort out equal the about difficult show points.

Research additional sources connected Saved Procedures and SQL Shack to deepen your knowing and better your SQL Server abilities. Cheque retired this article connected uncovering dilatory moving queries.

FAQ

Q: Wherefore is my question accelerated once executed straight however dilatory successful a saved process?

A: This is frequently owed to parameter sniffing, outdated statistic, oregon program caching points. The question optimizer creates a program primarily based connected the first parameters, which mightiness beryllium inefficient for consequent calls with antithetic parameters.

Optimizing your saved procedures is an ongoing procedure. By staying knowledgeable astir champion practices and using disposable instruments, you tin guarantee businesslike execution and a responsive exertion. Don’t fto dilatory saved procedures hinder your exertion’s show. Return the steps outlined supra to place and code the base causes, and unlock the afloat possible of your SQL Server database.

Question & Answer :
Abbreviated Interpretation

A question runs accelerated:

State @SessionGUID uniqueidentifier Fit @SessionGUID = 'BCBA333C-B6A1-4155-9833-C495F22EA908' Choice * FROM Report_Opener Wherever SessionGUID = @SessionGUID Command BY CurrencyTypeOrder, Fertile 

subtree outgo: zero.502

However placing the aforesaid SQL successful a saved process runs dilatory, and with a wholly antithetic execution program

Make Process dbo.ViewOpener @SessionGUID uniqueidentifier Arsenic Choice * FROM Report_Opener Wherever SessionGUID = @SessionGUID Command BY CurrencyTypeOrder, Fertile EXECUTE ViewOpener @SessionGUID 

Subtree outgo: 19.2

Agelong Interpretation

I’ve tally

sp_recompile ViewOpener 

And it inactive runs the aforesaid (severely), and I’ve besides modified the saved process to

Make Process dbo.ViewOpener @SessionGUID uniqueidentifier Arsenic Choice *, 'recompile delight' FROM Report_Opener Wherever SessionGUID = @SessionGUID Command BY CurrencyTypeOrder, Fertile 

And backmost once more, attempting to truly device it into recompiling.

I’ve dropped and recreated the saved process successful command to acquire it to make a fresh program.

I’ve tried forcing recompiles, and forestall parameter sniffing, by utilizing a decoy adaptable:

Make Process dbo.ViewOpener @SessionGUID uniqueidentifier Arsenic State @SessionGUIDbitch uniqueidentifier Fit @SessionGUIDbitch = @SessionGUID Choice * FROM Report_Opener Wherever SessionGUID = @SessionGUIDbitch Command BY CurrencyTypeOrder, Fertile 

I’ve besides tried defining the saved process WITH RECOMPILE:

Make Process dbo.ViewOpener @SessionGUID uniqueidentifier WITH RECOMPILE Arsenic Choice * FROM Report_Opener Wherever SessionGUID = @SessionGUID Command BY CurrencyTypeOrder, Fertile 

Truthful that it’s program is ne\’er cached, and I’ve tried forcing a recompile astatine execute:

EXECUTE ViewOpener @SessionGUID WITH RECOMPILE 

Which didn’t aid.

I’ve tried changing the process to dynamic SQL:

Make Process dbo.ViewOpener @SessionGUID uniqueidentifier WITH RECOMPILE Arsenic State @SQLString NVARCHAR(500) Fit @SQLString = N'Choice * FROM Report_OpenerTest Wherever SessionGUID = @SessionGUID Command BY CurrencyTypeOrder, Fertile' EXECUTE sp_executesql @SQLString, N'@SessionGUID uniqueidentifier', @SessionGUID 

Which didn’t aid.

The entity “Report_Opener” is a position, which is not listed. The position lone references underlying tables. Nary array incorporates computed columns, listed oregon other.

For the hellhole of it I tried creating the position with

Fit ANSI_NULLS Connected Fit QUOTED_IDENTIFER Connected 

That didn’t hole it.

However is it that

  • the question is accelerated
  • transferring the question to a position, and deciding on from the position is accelerated
  • deciding on from the position from a saved process is 40x slower?

I tried shifting the explanation of the position straight into the saved process (violating three concern guidelines, and breaking an crucial encapsulation), and that makes it lone astir 6x slower.

Wherefore is the saved process interpretation truthful dilatory? What tin perchance relationship for SQL Server moving advertisement-hoc SQL sooner than a antithetic benignant of advertisement-hoc SQL?

I’d truly instead not

  • embed the SQL successful codification

  • alteration the codification astatine each

    Microsoft SQL Server 2000 - eight.00.2050 (Intel X86) Mar 7 2008 21:29:fifty six Copyright (c) 1988-2003 Microsoft Corp Modular Variation connected Home windows NT 5.2 (Physique 3790: Work Battalion 2) 
    

However what tin relationship for SQL Server being incapable to tally arsenic accelerated arsenic SQL Sever moving a question, if not parameter sniffing.


My adjacent effort volition beryllium to person StoredProcedureA call StoredProcedureB call StoredProcedureC call StoredProcedureD to question the position.

And failing that, person the saved process call a saved process, call a UDF, call a UDF, call a saved process, call a UDF to question the position.


To sum ahead, the pursuing tally accelerated from QA, however dilatory once option into a saved process:

The first:

--Runs good extracurricular of a saved process Choice * FROM Report_OpenerTest Wherever SessionGUID = @SessionGUID Command BY CurrencyTypeOrder, Fertile 

sp_executesql:

--Runs good extracurricular of a saved process State @SQLString NVARCHAR(500) Fit @SQLString = N'Choice * FROM Report_OpenerTest Wherever SessionGUID = @SessionGUID Command BY CurrencyTypeOrder, Fertile' EXECUTE sp_executesql @SQLString, N'@SessionGUID uniqueidentifier', @SessionGUID 

EXEC(@sql):

--Runs good extracurricular of a saved process State @sql NVARCHAR(500) Fit @sql = N'Choice * FROM Report_OpenerTest Wherever SessionGUID = '''+Formed(@SessionGUID Arsenic varchar(50))+''' Command BY CurrencyTypeOrder, Fertile' EXEC(@sql) 

Execution Plans

The bully program:

|--Kind(Command BY:([Expr1020] ASC, [Currencies].[Fertile] ASC)) |--Compute Scalar(Specify:([Expr1020]=If ([Currencies].[CurrencyType]='ctCanadianCash') past 1 other If ([Currencies].[CurrencyType]='ctMiscellaneous') past 2 other If ([Currencies].[CurrencyType]='ctTokens') past three other If ([Currencies].[CurrencyType] |--Nested Loops(Near Outer Articulation, OUTER REFERENCES:([Openers].[OpenerGUID])) |--Filter(Wherever:((([Currencies].[IsActive]<>zero AND [Currencies].[OnOpener]<>zero) AND ((((((([Currencies].[CurrencyType]='ctUSCoin' Oregon [Currencies].[CurrencyType]='ctMiscellaneousUS') Oregon [Currencies].[CurrencyType]='ctUSCash') Oregon [Currencies]. | |--Nested Loops(Near Outer Articulation, OUTER REFERENCES:([Currencies].[CurrencyGUID], [Openers].[OpenerGUID]) WITH PREFETCH) | |--Nested Loops(Near Outer Articulation) | | |--Bookmark Lookup(BOOKMARK:([Bmk1016]), Entity:([GrobManagementSystemLive].[dbo].[Home windows])) | | | |--Nested Loops(Interior Articulation, OUTER REFERENCES:([Openers].[WindowGUID])) | | | |--Bookmark Lookup(BOOKMARK:([Bmk1014]), Entity:([GrobManagementSystemLive].[dbo].[Openers])) | | | | |--Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[Openers].[IX_Openers_SessionGUID]), Movement:([Openers].[SessionGUID]=[@SessionGUID]) ORDERED Guardant) | | | |--Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[Home windows].[IX_Windows]), Movement:([Home windows].[WindowGUID]=[Openers].[WindowGUID]) ORDERED Guardant) | | |--Clustered Scale Scan(Entity:([GrobManagementSystemLive].[dbo].[Currencies].[IX_Currencies_CurrencyType])) | |--Clustered Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[OpenerDetails].[IX_OpenerDetails_OpenerGUIDCurrencyGUID]), Movement:([OpenerDetails].[OpenerGUID]=[Openers].[OpenerGUID] AND [OpenerDetails].[CurrencyGUID]=[Currenc |--Hash Lucifer(Cache, HASH:([Openers].[OpenerGUID]), RESIDUAL:([Openers].[OpenerGUID]=[Openers].[OpenerGUID])) |--Watercourse Mixture(Specify:([Expr1006]=SUM(If (((([Currencies].[CurrencyType]='ctMiscellaneous' Oregon [Currencies].[CurrencyType]='ctTokens') Oregon [Currencies].[CurrencyType]='ctChips') Oregon [Currencies].[CurrencyType]='ctCanadianCoin') Oregon [ |--Nested Loops(Interior Articulation, OUTER REFERENCES:([OpenerDetails].[CurrencyGUID]) WITH PREFETCH) |--Nested Loops(Interior Articulation) | |--Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[Openers].[IX_Openers_OneOpenerPerSession]), Movement:([Openers].[OpenerGUID]=[Openers].[OpenerGUID]) ORDERED Guardant) | |--Clustered Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[OpenerDetails].[IX_OpenerDetails_OpenerGUIDCurrencyGUID]), Movement:([OpenerDetails].[OpenerGUID]=[Openers].[OpenerGUID]) ORDERED Guardant) |--Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[Currencies].[PK_Currencies_CurrencyGUID]), Movement:([Currencies].[CurrencyGUID]=[OpenerDetails].[CurrencyGUID]) ORDERED Guardant) 

The atrocious program

|--Kind(Command BY:([Expr1020] ASC, [Currencies].[Fertile] ASC)) |--Compute Scalar(Specify:([Expr1020]=If ([Currencies].[CurrencyType]='ctCanadianCash') past 1 other If ([Currencies].[CurrencyType]='ctMiscellaneous') past 2 other If ([Currencies].[CurrencyType]='ctTokens') past three other If ([Currencies].[Foreign money |--Nested Loops(Near Outer Articulation, OUTER REFERENCES:([Openers].[OpenerGUID])) |--Filter(Wherever:((([Currencies].[IsActive]<>zero AND [Currencies].[OnOpener]<>zero) AND ((((((([Currencies].[CurrencyType]='ctUSCoin' Oregon [Currencies].[CurrencyType]='ctMiscellaneousUS') Oregon [Currencies].[CurrencyType]='ctUSCash') Oregon [Currenc | |--Nested Loops(Near Outer Articulation, OUTER REFERENCES:([Currencies].[CurrencyGUID], [Openers].[OpenerGUID]) WITH PREFETCH) | |--Filter(Wherever:([Openers].[SessionGUID]=[@SessionGUID])) | | |--Concatenation | | |--Nested Loops(Near Outer Articulation) | | | |--Array Spool | | | | |--Hash Lucifer(Interior Articulation, HASH:([Home windows].[WindowGUID])=([Openers].[WindowGUID]), RESIDUAL:([Home windows].[WindowGUID]=[Openers].[WindowGUID])) | | | | |--Clustered Scale Scan(Entity:([GrobManagementSystemLive].[dbo].[Home windows].[IX_Windows_CageGUID])) | | | | |--Array Scan(Entity:([GrobManagementSystemLive].[dbo].[Openers])) | | | |--Array Spool | | | |--Clustered Scale Scan(Entity:([GrobManagementSystemLive].[dbo].[Currencies].[IX_Currencies_CurrencyType])) | | |--Compute Scalar(Specify:([Openers].[OpenerGUID]=NULL, [Openers].[SessionGUID]=NULL, [Home windows].[UseChipDenominations]=NULL)) | | |--Nested Loops(Near Anti Semi Articulation) | | |--Clustered Scale Scan(Entity:([GrobManagementSystemLive].[dbo].[Currencies].[IX_Currencies_CurrencyType])) | | |--Line Number Spool | | |--Array Spool | |--Clustered Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[OpenerDetails].[IX_OpenerDetails_OpenerGUIDCurrencyGUID]), Movement:([OpenerDetails].[OpenerGUID]=[Openers].[OpenerGUID] AND [OpenerDetails].[CurrencyGUID]=[Cu |--Hash Lucifer(Cache, HASH:([Openers].[OpenerGUID]), RESIDUAL:([Openers].[OpenerGUID]=[Openers].[OpenerGUID])) |--Watercourse Combination(Specify:([Expr1006]=SUM([partialagg1034]), [Expr1007]=SUM([partialagg1035]), [Expr1008]=SUM([partialagg1036]), [Expr1009]=SUM([partialagg1037]), [Expr1010]=SUM([partialagg1038]), [Expr1011]=SUM([partialagg1039] |--Nested Loops(Interior Articulation) |--Watercourse Combination(Specify:([partialagg1034]=SUM(If (((([Currencies].[CurrencyType]='ctMiscellaneous' Oregon [Currencies].[CurrencyType]='ctTokens') Oregon [Currencies].[CurrencyType]='ctChips') Oregon [Currencies].[CurrencyType]=' | |--Nested Loops(Interior Articulation, OUTER REFERENCES:([OpenerDetails].[CurrencyGUID]) WITH PREFETCH) | |--Clustered Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[OpenerDetails].[IX_OpenerDetails_OpenerGUIDCurrencyGUID]), Movement:([OpenerDetails].[OpenerGUID]=[Openers].[OpenerGUID]) ORDERED Guardant) | |--Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[Currencies].[PK_Currencies_CurrencyGUID]), Movement:([Currencies].[CurrencyGUID]=[OpenerDetails].[CurrencyGUID]) ORDERED Guardant) |--Scale Movement(Entity:([GrobManagementSystemLive].[dbo].[Openers].[IX_Openers_OneOpenerPerSession]), Movement:([Openers].[OpenerGUID]=[Openers].[OpenerGUID]) ORDERED Guardant) 

The atrocious-1 is anxious spooling 6 cardinal rows; the another 1 isn’t.

Line: This isn’t a motion astir tuning a question. I person a question that runs lightning accelerated. I conscionable privation SQL Server to tally accelerated from a saved process.

I had the aforesaid job arsenic the first poster however the quoted reply did not lick the job for maine. The question inactive ran truly dilatory from a saved process.

I recovered different reply present “Parameter Sniffing”, Acknowledgment Omnibuzz. Boils behind to utilizing “section Variables” successful your saved process queries, however publication the first for much knowing, it’s a large compose ahead. e.g.

Dilatory manner:

Make Process GetOrderForCustomers(@CustID varchar(20)) Arsenic Statesman Choice * FROM orders Wherever customerid = @CustID Extremity 

Accelerated manner:

Make Process GetOrderForCustomersWithoutPS(@CustID varchar(20)) Arsenic Statesman State @LocCustID varchar(20) Fit @LocCustID = @CustID Choice * FROM orders Wherever customerid = @LocCustID Extremity 

Doing this lowered my execution clip from 5+ minutes to astir 6-7 seconds.