Database
 sql >> डेटाबेस >  >> RDS >> Database

टी-एसक्यूएल मंगलवार #64 :एक ट्रिगर या कई?

यह महीने का वह मंगलवार है - आप जानते हैं, जब ब्लॉगर ब्लॉक पार्टी जिसे टी-एसक्यूएल मंगलवार के रूप में जाना जाता है, होता है। इस महीने यह रस थॉमस (@SQLJudo) द्वारा होस्ट किया गया है, और विषय है, "सभी ट्यूनर और गियर प्रमुखों को कॉल करना।" मैं यहां एक प्रदर्शन-संबंधी समस्या का इलाज करने जा रहा हूं, हालांकि मैं क्षमा चाहता हूं कि यह पूरी तरह से उन दिशानिर्देशों के अनुरूप नहीं हो सकता है जो रसेल ने अपने निमंत्रण में निर्धारित किए हैं (मैं संकेत, ट्रेस झंडे या योजना गाइड का उपयोग नहीं करने जा रहा हूं) ।

पिछले हफ्ते SQLBits में, मैंने ट्रिगर्स पर एक प्रेजेंटेशन दिया, और मेरे अच्छे दोस्त और साथी MVP Erland Somarskog उपस्थित हुए। एक बिंदु पर मैंने सुझाव दिया कि एक टेबल पर एक नया ट्रिगर बनाने से पहले, आपको यह देखने के लिए जांच करनी चाहिए कि क्या कोई ट्रिगर पहले से मौजूद है, और अतिरिक्त ट्रिगर जोड़ने के बजाय तर्क को संयोजित करने पर विचार करें। मेरे कारण मुख्य रूप से कोड रखरखाव के लिए थे, लेकिन प्रदर्शन के लिए भी। एरलैंड ने पूछा कि क्या मैंने कभी यह देखने के लिए परीक्षण किया था कि क्या एक ही कार्रवाई के लिए कई ट्रिगर्स आग लगने में कोई अतिरिक्त ओवरहेड था, और मुझे यह स्वीकार करना पड़ा कि, नहीं, मैंने कुछ भी व्यापक नहीं किया था। तो मैं अभी ऐसा करने जा रहा हूँ।

AdventureWorks2014 में, मैंने टेबल का एक सरल सेट बनाया जो मूल रूप से sys.all_objects का प्रतिनिधित्व करता है। (~2,700 पंक्तियाँ) और sys.all_columns (~9,500 पंक्तियाँ)। मैं दोनों तालिकाओं को अद्यतन करने के लिए विभिन्न दृष्टिकोणों के कार्यभार पर प्रभाव को मापना चाहता था - अनिवार्य रूप से आपके पास कॉलम तालिका अपडेट करने वाले उपयोगकर्ता हैं, और आप एक ही तालिका में एक अलग कॉलम और ऑब्जेक्ट तालिका में कुछ कॉलम अपडेट करने के लिए ट्रिगर का उपयोग करते हैं।

  • T1:आधार रेखा :मान लें कि आप संग्रहीत कार्यविधि के माध्यम से सभी डेटा एक्सेस को नियंत्रित कर सकते हैं; इस मामले में, दोनों तालिकाओं के खिलाफ अद्यतन सीधे किया जा सकता है, ट्रिगर्स की कोई आवश्यकता नहीं है। (यह वास्तविक दुनिया में व्यावहारिक नहीं है, क्योंकि आप तालिकाओं तक सीधे पहुंच को विश्वसनीय रूप से प्रतिबंधित नहीं कर सकते हैं।)
  • T2:अन्य तालिका के विरुद्ध एकल ट्रिगर :मान लें कि आप प्रभावित तालिका के विरुद्ध अद्यतन कथन को नियंत्रित कर सकते हैं और अन्य स्तंभ जोड़ सकते हैं, लेकिन द्वितीयक तालिका के अद्यतनों को ट्रिगर के साथ कार्यान्वित करने की आवश्यकता है। हम तीनों कॉलम को एक स्टेटमेंट के साथ अपडेट करेंगे।
  • T3:दोनों तालिकाओं के विरुद्ध एकल ट्रिगर :इस मामले में, हमारे पास दो कथनों के साथ एक ट्रिगर है, एक जो प्रभावित तालिका में दूसरे कॉलम को अपडेट करता है, और एक जो द्वितीयक तालिका में सभी तीन कॉलम को अपडेट करता है।
  • T4:दोनों तालिकाओं के विरुद्ध एकल ट्रिगर :T3 की तरह, लेकिन इस बार, हमारे पास चार कथनों के साथ एक ट्रिगर है, एक जो प्रभावित तालिका में दूसरे कॉलम को अपडेट करता है, और द्वितीयक तालिका में अपडेट किए गए प्रत्येक कॉलम के लिए एक स्टेटमेंट। यदि समय के साथ आवश्यकताओं को जोड़ा जाता है और प्रतिगमन परीक्षण के संदर्भ में एक अलग कथन को सुरक्षित माना जाता है, तो इसे इस तरह से संभाला जा सकता है।
  • T5:दो ट्रिगर :एक ट्रिगर केवल प्रभावित तालिका को अपडेट करता है; दूसरा द्वितीयक तालिका में तीन स्तंभों को अद्यतन करने के लिए एकल कथन का उपयोग करता है। यदि अन्य ट्रिगर्स पर ध्यान नहीं दिया जाता है या यदि उन्हें संशोधित करना प्रतिबंधित है, तो इसे इसी तरह से किया जा सकता है।
  • T6:चार ट्रिगर :एक ट्रिगर केवल प्रभावित तालिका को अपडेट करता है; अन्य तीन द्वितीयक तालिका में प्रत्येक स्तंभ को अद्यतन करते हैं। दोबारा, यदि आप नहीं जानते कि अन्य ट्रिगर मौजूद हैं, या यदि आप प्रतिगमन चिंताओं के कारण अन्य ट्रिगर्स को छूने से डरते हैं, तो यह उसी तरह से किया जा सकता है।

यहां वह स्रोत डेटा है जिसके साथ हम काम कर रहे हैं:

-- sys.all_objects:
SELECT * INTO dbo.src FROM sys.all_objects;
CREATE UNIQUE CLUSTERED INDEX x ON dbo.src([object_id]);
GO
 
-- sys.all_columns:
SELECT * INTO dbo.tr1 FROM sys.all_columns;
CREATE UNIQUE CLUSTERED INDEX x ON dbo.tr1([object_id], column_id);
-- repeat 5 times: tr2, tr3, tr4, tr5, tr6

अब, 6 परीक्षणों में से प्रत्येक के लिए, हम अपने अपडेट 1,000 बार चलाने जा रहे हैं, और समय की लंबाई को मापेंगे

T1:बेसलाइन

यह वह परिदृश्य है जहां हम ट्रिगर से बचने के लिए पर्याप्त भाग्यशाली हैं (फिर से, बहुत यथार्थवादी नहीं)। इस मामले में, हम इस बैच की रीडिंग और अवधि को मापेंगे। मैंने /*real*/ डाला क्वेरी टेक्स्ट में ताकि मैं केवल इन कथनों के आंकड़े आसानी से खींच सकूं, न कि ट्रिगर्स के भीतर से कोई भी कथन, क्योंकि अंततः मेट्रिक्स ट्रिगर्स को लागू करने वाले कथनों तक रोल अप करते हैं। यह भी ध्यान दें कि मेरे द्वारा किए जा रहे वास्तविक अपडेट का वास्तव में कोई मतलब नहीं है, इसलिए इस बात पर ध्यान न दें कि मैं सर्वर/इंस्टेंस नाम और ऑब्जेक्ट के principal_id पर कॉलेशन सेट कर रहा हूं। वर्तमान सत्र के session_id . पर ।

UPDATE /*real*/ dbo.tr1 SET name += N'',
  collation_name = @@SERVERNAME
  WHERE name LIKE '%s%';
 
UPDATE /*real*/ s SET modify_date = GETDATE(), is_ms_shipped = 0, principal_id = @@SPID
  FROM dbo.src AS s
  INNER JOIN dbo.tr1 AS t
  ON s.[object_id] = t.[object_id]
  WHERE t.name LIKE '%s%';
 
GO 1000

T2:सिंगल ट्रिगर

इसके लिए हमें निम्नलिखित सरल ट्रिगर की आवश्यकता है, जो केवल dbo.src को अपडेट करता है :

CREATE TRIGGER dbo.tr_tr2
ON dbo.tr2
AFTER UPDATE
AS
BEGIN
  SET NOCOUNT ON;
  UPDATE s SET modify_date = GETDATE(), is_ms_shipped = 0, principal_id = SUSER_ID()
    FROM dbo.src AS s 
	INNER JOIN inserted AS i
	ON s.[object_id] = i.[object_id];
END
GO

तब हमारे बैच को केवल प्राथमिक तालिका में दो स्तंभों को अपडेट करने की आवश्यकता होती है:

UPDATE /*real*/ dbo.tr2 SET name += N'', collation_name = @@SERVERNAME
  WHERE name LIKE '%s%';
GO 1000

T3:दोनों तालिकाओं के विरुद्ध एकल ट्रिगर

इस परीक्षण के लिए, हमारा ट्रिगर इस तरह दिखता है:

CREATE TRIGGER dbo.tr_tr3
ON dbo.tr3
AFTER UPDATE
AS
BEGIN
  SET NOCOUNT ON;
  UPDATE t SET collation_name = @@SERVERNAME
    FROM dbo.tr3 AS t
	INNER JOIN inserted AS i
	ON t.[object_id] = i.[object_id];
 
  UPDATE s SET modify_date = GETDATE(), is_ms_shipped = 0, principal_id = @@SPID
    FROM dbo.src AS s
    INNER JOIN inserted AS i
    ON s.[object_id] = i.[object_id];
END
GO

और अब हम जिस बैच का परीक्षण कर रहे हैं, उसे केवल प्राथमिक तालिका में मूल कॉलम को अपडेट करना है; दूसरे को ट्रिगर द्वारा नियंत्रित किया जाता है:

UPDATE /*real*/ dbo.tr3 SET name += N''
  WHERE name LIKE '%s%';
GO 1000

T4:दोनों तालिकाओं के विरुद्ध एकल ट्रिगर

यह बिल्कुल T3 जैसा है, लेकिन अब ट्रिगर के चार कथन हैं:

CREATE TRIGGER dbo.tr_tr4
ON dbo.tr4
AFTER UPDATE
AS
BEGIN
  SET NOCOUNT ON;
  UPDATE t SET collation_name = @@SERVERNAME
    FROM dbo.tr4 AS t
	INNER JOIN inserted AS i
	ON t.[object_id] = i.[object_id];
 
  UPDATE s SET modify_date = GETDATE()
    FROM dbo.src AS s
    INNER JOIN inserted AS i
    ON s.[object_id] = i.[object_id];
 
  UPDATE s SET is_ms_shipped = 0
    FROM dbo.src AS s
    INNER JOIN inserted AS i
    ON s.[object_id] = i.[object_id];
 
  UPDATE s SET principal_id = @@SPID
    FROM dbo.src AS s
    INNER JOIN inserted AS i
    ON s.[object_id] = i.[object_id];
END
GO
डाला गया

परीक्षण बैच अपरिवर्तित है:

UPDATE /*real*/ dbo.tr4 SET name += N''
  WHERE name LIKE '%s%';
GO 1000

T5:दो ट्रिगर

यहां हमारे पास प्राथमिक तालिका को अपडेट करने के लिए एक ट्रिगर है, और द्वितीयक तालिका को अपडेट करने के लिए एक ट्रिगर है:

CREATE TRIGGER dbo.tr_tr5_1
ON dbo.tr5
AFTER UPDATE
AS
BEGIN
  SET NOCOUNT ON;
  UPDATE t SET collation_name = @@SERVERNAME
    FROM dbo.tr5 AS t
	INNER JOIN inserted AS i
	ON t.[object_id] = i.[object_id];
END
GO
 
CREATE TRIGGER dbo.tr_tr5_2
ON dbo.tr5
AFTER UPDATE
AS
BEGIN
  SET NOCOUNT ON;
  UPDATE s SET modify_date = GETDATE(), is_ms_shipped = 0, principal_id = @@SPID
    FROM dbo.src AS s
    INNER JOIN inserted AS i
    ON s.[object_id] = i.[object_id];
END
GO

परीक्षण बैच फिर से बहुत ही बुनियादी है:

UPDATE /*real*/ dbo.tr5 SET name += N''
  WHERE name LIKE '%s%';
GO 1000

T6:चार ट्रिगर

इस बार हमारे पास प्रभावित होने वाले प्रत्येक कॉलम के लिए एक ट्रिगर है; एक प्राथमिक तालिका में, और तीन माध्यमिक तालिकाओं में।

CREATE TRIGGER dbo.tr_tr6_1
ON dbo.tr6
AFTER UPDATE
AS
BEGIN
  SET NOCOUNT ON;
  UPDATE t SET collation_name = @@SERVERNAME
    FROM dbo.tr6 AS t
    INNER JOIN inserted AS i
    ON t.[object_id] = i.[object_id];
END
GO
 
CREATE TRIGGER dbo.tr_tr6_2
ON dbo.tr6
AFTER UPDATE
AS
BEGIN
  SET NOCOUNT ON;
  UPDATE s SET modify_date = GETDATE()
    FROM dbo.src AS s
    INNER JOIN inserted AS i
    ON s.[object_id] = i.[object_id];
END
GO
 
CREATE TRIGGER dbo.tr_tr6_3
ON dbo.tr6
AFTER UPDATE
AS
BEGIN
  SET NOCOUNT ON;
  UPDATE s SET is_ms_shipped = 0
    FROM dbo.src AS s
    INNER JOIN inserted AS i
    ON s.[object_id] = i.[object_id];
END
GO
 
CREATE TRIGGER dbo.tr_tr6_4
ON dbo.tr6
AFTER UPDATE
AS
BEGIN
  SET NOCOUNT ON;
  UPDATE s SET principal_id = @@SPID
    FROM dbo.src AS s
    INNER JOIN inserted AS i
    ON s.[object_id] = i.[object_id];
END
GO
डाला गया

और परीक्षण बैच:

UPDATE /*real*/ dbo.tr6 SET name += N''
  WHERE name LIKE '%s%';
GO 1000

कार्यभार के प्रभाव को मापना

अंत में, मैंने sys.dm_exec_query_stats . के विरुद्ध एक सरल क्वेरी लिखी प्रत्येक परीक्षण के लिए पठन और अवधि मापने के लिए:

SELECT 
  [cmd] = SUBSTRING(t.text, CHARINDEX(N'U', t.text), 23), 
  avg_elapsed_time = total_elapsed_time / execution_count * 1.0,
  total_logical_reads
FROM sys.dm_exec_query_stats AS s 
CROSS APPLY sys.dm_exec_sql_text(s.sql_handle) AS t
WHERE t.text LIKE N'%UPDATE /*real*/%'
ORDER BY cmd;

परिणाम

मैंने 10 बार परीक्षण चलाए, परिणाम एकत्र किए, और सब कुछ औसत किया। यहां बताया गया है कि यह कैसे टूट गया:

परीक्षा/बैच औसत अवधि
(माइक्रोसेकंड)
कुल पढ़ा गया
(8K पेज)
T1 :अद्यतन /*असली*/ dbo.tr1 … 22,608 205,134
T2 :अद्यतन /*असली*/ dbo.tr2 … 32,749 11,331,628
T3 :अद्यतन /*असली*/ dbo.tr3 … 72,899 22,838,308
T4 :अद्यतन /*असली*/ dbo.tr4 … 78,372 44,463,275
T5 :अद्यतन /*असली*/ dbo.tr5 … 88,563 41,514,778
T6 :अद्यतन /*असली*/ dbo.tr6 … 127,079 100,330,753


और यहां अवधि का चित्रमय प्रतिनिधित्व है:

निष्कर्ष

यह स्पष्ट है कि, इस मामले में, लागू होने वाले प्रत्येक ट्रिगर के लिए कुछ पर्याप्त ओवरहेड है - इन सभी बैचों ने अंततः समान पंक्तियों को प्रभावित किया, लेकिन कुछ मामलों में एक ही पंक्तियों को कई बार छुआ गया था। जब एक ही पंक्ति को एक से अधिक बार स्पर्श नहीं किया जाता है, तो अंतर को मापने के लिए मैं शायद आगे अनुवर्ती परीक्षण करूंगा - एक अधिक जटिल स्कीमा, शायद, जहां हर बार 5 या 10 अन्य तालिकाओं को छूना पड़ता है, और ये अलग-अलग कथन हो सकते हैं एक ट्रिगर में या एकाधिक में। मेरा अनुमान है कि ओवरहेड अंतर समवर्ती जैसी चीजों और ट्रिगर के ऊपरी हिस्से की तुलना में प्रभावित पंक्तियों की संख्या से अधिक संचालित होंगे - लेकिन हम देखेंगे।

स्वयं डेमो आज़माना चाहते हैं? स्क्रिप्ट यहाँ से डाउनलोड करें।


  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. SQL में प्राथमिक कुंजी कैसे बनाएं

  2. डेटाबेस रीऑर्ग्स - वे क्यों मायने रखते हैं?

  3. SQL में वर्णानुक्रम में ऑर्डर कैसे करें

  4. SQL में स्कीमा क्या है और इसे कैसे बनाया जाता है?

  5. DBCC_OBJECT_METADATA कुंडी