SQL सर्वर की दुनिया में, दो प्रकार के लोग होते हैं:वे जो अपनी सभी वस्तुओं को उपसर्ग करना पसंद करते हैं, और जो नहीं करते हैं। पूर्व समूह को आगे दो श्रेणियों में विभाजित किया गया है:वे जो संग्रहीत कार्यविधियों को sp_ . के साथ उपसर्ग करते हैं , और जो अन्य उपसर्ग चुनते हैं (जैसे usp_ या proc_ ) sp_ . से बचने के लिए लंबे समय से अनुशंसा की जाती रही है उपसर्ग, दोनों प्रदर्शन कारणों से, और अस्पष्टता या टकराव से बचने के लिए यदि आप ऐसा नाम चुनते हैं जो सिस्टम कैटलॉग में पहले से मौजूद है। टकराव निश्चित रूप से अभी भी एक मुद्दा है, लेकिन यह मानते हुए कि आपने अपने ऑब्जेक्ट नाम की जांच की है, क्या यह अभी भी एक प्रदर्शन समस्या है?
TL;DR संस्करण:हाँ।
sp_ उपसर्ग अभी भी नहीं-नहीं है। लेकिन इस पोस्ट में मैं समझाऊंगा कि क्यों, SQL सर्वर 2012 आपको यह विश्वास दिलाने के लिए कैसे प्रेरित कर सकता है कि यह चेतावनी सलाह अब लागू नहीं होती है, और इस नामकरण सम्मेलन को चुनने के कुछ अन्य संभावित दुष्प्रभाव।
sp_ में क्या समस्या है?
sp_ उपसर्ग का मतलब यह नहीं है कि आपको क्या लगता है कि यह क्या करता है:ज्यादातर लोग सोचते हैं sp "संग्रहीत प्रक्रिया" के लिए खड़ा है जब वास्तव में इसका अर्थ "विशेष" है। संग्रहीत कार्यविधियाँ (साथ ही तालिकाएँ और दृश्य) एक sp_ . के साथ मास्टर में संग्रहीत हैं उपसर्ग किसी भी डेटाबेस से उचित संदर्भ के बिना पहुंच योग्य हैं (यह मानते हुए कि स्थानीय संस्करण मौजूद नहीं है)। यदि प्रक्रिया को सिस्टम ऑब्जेक्ट के रूप में चिह्नित किया गया है (sp_MS_marksystemobject . का उपयोग करके) (एक गैर-दस्तावेजी और असमर्थित सिस्टम प्रक्रिया जो is_ms_shipped . सेट करती है से 1), तो मास्टर में प्रक्रिया कॉलिंग डेटाबेस के संदर्भ में निष्पादित होगी। आइए एक सरल उदाहरण देखें:
CREATE DATABASE sp_test;
GO
USE sp_test;
GO
CREATE TABLE dbo.foo(id INT);
GO
USE master;
GO
CREATE PROCEDURE dbo.sp_checktable
AS
SELECT DB_NAME(), name
FROM sys.tables WHERE name = N'foo';
GO
USE sp_test;
GO
EXEC dbo.sp_checktable; -- runs but returns 0 results
GO
EXEC master..sp_MS_marksystemobject N'dbo.sp_checktable';
GO
EXEC dbo.sp_checktable; -- runs and returns results
GO परिणाम:
(0 row(s) affected) sp_test foo (1 row(s) affected)
प्रदर्शन समस्या इस तथ्य से आती है कि मास्टर को समकक्ष संग्रहीत प्रक्रिया के लिए जांचा जा सकता है, इस पर निर्भर करता है कि प्रक्रिया का स्थानीय संस्करण है या नहीं, और वास्तव में मास्टर में समकक्ष वस्तु है या नहीं। इससे अतिरिक्त मेटाडेटा ओवरहेड के साथ-साथ एक अतिरिक्त SP:CacheMiss . हो सकता है प्रतिस्पर्धा। सवाल यह है कि क्या यह ओवरहेड मूर्त है।
तो आइए एक परीक्षण डेटाबेस में एक बहुत ही सरल प्रक्रिया पर विचार करें:
CREATE DATABASE sp_prefix; GO USE sp_prefix; GO CREATE PROCEDURE dbo.sp_something AS BEGIN SELECT 'sp_prefix', DB_NAME(); END GO
और मास्टर में समकक्ष प्रक्रियाएं:
USE master; GO CREATE PROCEDURE dbo.sp_something AS BEGIN SELECT 'master', DB_NAME(); END GO EXEC sp_MS_marksystemobject N'sp_something';
CacheMiss :तथ्य या कल्पना?
यदि हम अपने परीक्षण डेटाबेस से एक त्वरित परीक्षण चलाते हैं, तो हम देखते हैं कि इन संग्रहीत प्रक्रियाओं को निष्पादित करने से वास्तव में कभी भी मास्टर से संस्करणों का आह्वान नहीं होगा, भले ही हम ठीक से डेटाबेस- या स्कीमा-योग्यता प्रक्रिया (एक सामान्य गलत धारणा) या यदि हम चिह्नित करते हैं सिस्टम ऑब्जेक्ट के रूप में मास्टर संस्करण:
USE sp_prefix; GO EXEC sp_prefix.dbo.sp_something; GO EXEC dbo.sp_something; GO EXEC sp_something;
परिणाम:
sp_prefix sp_prefix sp_prefix sp_prefix sp_prefix sp_prefix
आइए एक क्विक ट्रेस भी चलाएंSP:CacheMiss है इवेंट:

हम देखते हैं CacheMiss तदर्थ बैच के लिए ईवेंट जो संग्रहीत कार्यविधि को कॉल करते हैं (चूंकि SQL सर्वर आमतौर पर उस बैच को कैशिंग करने में परेशानी नहीं करेगा जिसमें मुख्य रूप से प्रक्रिया कॉल शामिल हैं), लेकिन संग्रहीत कार्यविधि के लिए नहीं। sp_something . के साथ और बिना दोनों मास्टर में मौजूद प्रक्रिया (और जब यह मौजूद है, दोनों के साथ और इसके बिना सिस्टम ऑब्जेक्ट के रूप में चिह्नित किया जा रहा है), sp_something पर कॉल करता है उपयोगकर्ता डेटाबेस में कभी भी "गलती से" प्रक्रिया को मास्टर में कॉल न करें, और कभी भी कोई CacheMiss उत्पन्न न करें प्रक्रिया के लिए घटनाएँ।
यह SQL Server 2012 पर था। मैंने उपरोक्त समान परीक्षण SQL Server 2008 R2 पर दोहराए, और थोड़े भिन्न परिणाम पाए:

तो SQL Server 2008 R2 पर हम एक अतिरिक्त CacheMiss . देखते हैं घटना जो SQL सर्वर 2012 में नहीं होती है। यह सभी परिदृश्यों में होता है (कोई समकक्ष ऑब्जेक्ट मास्टर नहीं, मास्टर में एक ऑब्जेक्ट को सिस्टम ऑब्जेक्ट के रूप में चिह्नित किया जाता है, और मास्टर में एक ऑब्जेक्ट को सिस्टम ऑब्जेक्ट के रूप में चिह्नित नहीं किया जाता है)। तुरंत मैं उत्सुक था कि क्या इस अतिरिक्त घटना का प्रदर्शन पर कोई उल्लेखनीय प्रभाव पड़ेगा।
प्रदर्शन समस्या:तथ्य या कल्पना?
मैंने sp_ . के बिना एक अतिरिक्त प्रक्रिया बनाई कच्चे प्रदर्शन की तुलना करने के लिए उपसर्ग, CacheMiss एक तरफ:
USE sp_prefix; GO CREATE PROCEDURE dbo.proc_something AS BEGIN SELECT 'sp_prefix', DB_NAME(); END GO
तो sp_something . के बीच एकमात्र अंतर और proc_something . फिर मैंने EXEC sp_prefix.dbo.<procname> का उपयोग करके, उन्हें 1000 बार निष्पादित करने के लिए रैपर प्रक्रियाएं बनाईं। , EXEC dbo.<procname> और EXEC <procname> सिंटैक्स, मास्टर में रहने वाले समकक्ष संग्रहीत प्रक्रियाओं के साथ और सिस्टम ऑब्जेक्ट के रूप में चिह्नित, मास्टर में रहना लेकिन सिस्टम ऑब्जेक्ट के रूप में चिह्नित नहीं, और मास्टर में बिल्कुल नहीं रहना।
USE sp_prefix;
GO
CREATE PROCEDURE dbo.wrap_sp_3part
AS
BEGIN
DECLARE @i INT = 1;
WHILE @i <= 1000
BEGIN
EXEC sp_prefix.dbo.sp_something;
SET @i += 1;
END
END
GO
CREATE PROCEDURE dbo.wrap_sp_2part
AS
BEGIN
DECLARE @i INT = 1;
WHILE @i <= 1000
BEGIN
EXEC dbo.sp_something;
SET @i += 1;
END
END
GO
CREATE PROCEDURE dbo.wrap_sp_1part
AS
BEGIN
DECLARE @i INT = 1;
WHILE @i <= 1000
BEGIN
EXEC sp_something;
SET @i += 1;
END
END
GO
-- repeat for proc_something
SQL संतरी योजना एक्सप्लोरर के साथ प्रत्येक रैपर प्रक्रिया की रनटाइम अवधि को मापने के परिणाम दिखाते हैं कि sp_ का उपयोग करना उपसर्ग का लगभग सभी मामलों में औसत अवधि पर महत्वपूर्ण प्रभाव पड़ता है (और निश्चित रूप से औसतन):

