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

यह मिथक कि DROP और TRUNCATE TABLE नॉन-लॉगेड हैं

SQL सर्वर की दुनिया में एक स्थायी मिथक है कि दोनों DROP TABLE और TRUNCATE TABLE आदेश गैर-लॉग हैं।

वे नहीं हैं। वे दोनों पूरी तरह से लॉग इन हैं, लेकिन कुशलता से लॉग इन हैं।

आप इसे आसानी से अपने लिए साबित कर सकते हैं। परीक्षण डेटाबेस और तालिका सेट करने के लिए निम्न कोड चलाएँ, और दिखाएँ कि हमारी तालिका में 10,000 पंक्तियाँ हैं:

CREATE DATABASE [TruncateTest];
GO
 
ALTER DATABASE [TruncateTest] SET RECOVERY SIMPLE;
GO
 
USE [TruncateTest];
GO
 
CREATE TABLE [TestTable] (
    [c1]    INT IDENTITY,
    [c2]    CHAR (8000) DEFAULT 'a');
GO
 
SET NOCOUNT ON;
GO
 
INSERT INTO [TestTable] DEFAULT VALUES;
GO 10000
 
SELECT
    COUNT (*) AS N'RowCount'
FROM
    [TestTable];
GO

परिणाम:

रोकाउंट
————
10000

और फिर निम्न कोड, जो लेन-देन में तालिका को छोटा करता है और पंक्ति गणना की जांच करता है:

BEGIN TRAN;
GO
TRUNCATE TABLE [TestTable];
GO
 
SELECT
    COUNT (*) AS N'RowCount'
FROM
    [TestTable];
GO

परिणाम:

रोकाउंट
————
0

अब टेबल खाली है। लेकिन मैं लेन-देन को वापस ले सकता हूं और सभी डेटा को फिर से वापस रख सकता हूं:

ROLLBACK TRAN;
GO
 
SELECT
    COUNT (*) AS N'RowCount'
FROM
    [TestTable];
GO

परिणाम:

रोकाउंट
————
10000

स्पष्ट रूप से TRUNCATE ऑपरेशन लॉग होना चाहिए अन्यथा रोल बैक ऑपरेशन काम नहीं करेगा।

तो ग़लतफ़हमी कहाँ से आती है?

यह DROP . के व्यवहार से आता है और TRUNCATE बड़ी मेजों पर संचालन। वे लगभग तुरंत ही पूर्ण हो जाएंगे, और यदि आप fn_dblog का उपयोग करके लेन-देन लॉग में देखते हैं ठीक बाद में, आप केवल ऑपरेशन से उत्पन्न लॉग रिकॉर्ड की एक छोटी संख्या देखेंगे। वह छोटी संख्या तालिका के आकार को छोटा या गिराए जाने से संबंधित नहीं है, इसलिए ऐसा लगता है जैसे DROP और TRUNCATE संचालन गैर-लॉग हैं।

लेकिन वे पूरी तरह से लॉग इन हैं, जैसा कि मैंने ऊपर दिखाया है। तो संचालन के लिए लॉग रिकॉर्ड कहाँ हैं?

इसका उत्तर यह है कि लॉग रिकॉर्ड 'डिफर्ड ड्रॉप' नामक एक तंत्र द्वारा बनाया जाएगा, बस तुरंत नहीं, जिसे SQL Server 2000 SP3 में जोड़ा गया था।

जब कोई तालिका गिरा दी जाती है या काट दी जाती है, तो तालिका के लिए आवंटित सभी डेटा फ़ाइल पृष्ठों को हटा दिया जाना चाहिए। SQL Server 2000 SP3 से पहले इसके लिए तंत्र इस प्रकार था:

तालिका में आवंटित प्रत्येक सीमा के लिए

शुरू करें

सीमा पर एक विशिष्ट आवंटन लॉक प्राप्त करें



इसकी जांच करें प्रत्येक पृष्ठ के लिए पेज लॉक इस हद तक (एक्सक्लूसिव मोड में लॉक प्राप्त करें, और तुरंत इसे छोड़ दें, सुनिश्चित करें कि किसी और ने पेज लॉक नहीं किया है)



ऐसा न करें सीमा लॉक जारी करें, यह गारंटी देते हुए कि कोई और उस सीमा का उपयोग नहीं कर सकता



अगली सीमा तक ले जाएं

समाप्त करें

चूंकि ऑपरेशन के अंत तक सभी हद तक ताले रखे गए थे, और प्रत्येक लॉक में थोड़ी मात्रा में मेमोरी होती है, इसलिए लॉक मैनेजर के लिए मेमोरी से बाहर निकलना संभव था जब एक DROP या TRUNCATE एक बहुत बड़ी मेज का हुआ। कुछ SQL सर्वर ग्राहकों ने पाया कि वे SQL Server 2000 पर आउट-ऑफ-मेमोरी स्थितियों में भाग गए, क्योंकि टेबल बहुत बड़े हो गए और सिस्टम मेमोरी में वृद्धि को काफी हद तक पीछे छोड़ दिया।

आस्थगित-ड्रॉप तंत्र DROP . का अनुकरण करता है या TRUNCATE एक पृष्ठभूमि कार्य द्वारा बाद में प्रसंस्करण के लिए, तालिका के लिए आवंटन को हटाकर और उन्हें 'आस्थगित-ड्रॉप कतार' पर रखकर तुरंत ऑपरेशन पूरा करना। यह अनहुक-एंड-ट्रांसफर ऑपरेशन केवल मुट्ठी भर लॉग रिकॉर्ड बनाता है। यह वह ऑपरेशन है जो ऊपर दिए गए मेरे कोड उदाहरण में किया जा रहा है और रोल-बैक किया जा रहा है।
'डिफर्ड-ड्रॉप बैकग्राउंड टास्क' हर कुछ सेकंड में घूमता है और डिफर्ड-ड्रॉप क्यू पर सभी पेजों और विस्तार को छोटे में हटा देता है बैच, यह गारंटी देता है कि ऑपरेशन स्मृति से बाहर नहीं होगा। ये डीलोकेशन पूरी तरह से लॉग किए गए हैं, लेकिन याद रखें कि डेटा या इंडेक्स रिकॉर्ड से भरे पेज को डीलोकेट करने से उन रिकॉर्ड्स के अलग-अलग डिलीट लॉग नहीं होते हैं; इसके बजाय पूरे पृष्ठ को प्रासंगिक पीएफएस (पेज फ्री स्पेस) आवंटन बाइट-मैप में डीललोकेटेड के रूप में चिह्नित किया गया है।

SQL Server 2000 SP3 से आगे, जब आप DROP करते हैं या TRUNCATE तालिका में, आप केवल कुछ लॉग रिकॉर्ड जेनरेट होते हुए देखेंगे। यदि आप एक या दो मिनट प्रतीक्षा करते हैं, और फिर लेन-देन लॉग में फिर से देखते हैं, तो आप देखेंगे कि हजारों लॉग रिकॉर्ड आस्थगित-ड्रॉप ऑपरेशन द्वारा उत्पन्न किए गए हैं, प्रत्येक एक पृष्ठ या सीमा को हटा रहा है। ऑपरेशन पूरी तरह से और कुशलता से लॉग किया गया है।

हमारे द्वारा ऊपर बनाए गए परिदृश्य का उपयोग करते हुए, यहां एक उदाहरण दिया गया है:

CHECKPOINT;
GO
TRUNCATE TABLE [TestTable];
GO
SELECT
    COUNT (*) AS N'LogRecCount'
FROM
    fn_dblog (NULL, NULL);
GO

परिणाम:

LogRecCount

———–

25

जैसा कि आप देख सकते हैं, स्पष्ट रूप से कोई लॉग रिकॉर्ड नहीं हैं जो टेस्टटेबल तालिका में 10,000 पृष्ठों, प्लस 1,250 विस्तारों को हटाते हैं।

अगर मैं कुछ सेकंड प्रतीक्षा करता हूं, और फिर fn_dblog चलाता हूं कोड फिर से, मुझे मिलता है:

LogRecCount

———–

3811

आपको आश्चर्य हो सकता है कि कम से कम 10,000 लॉग रिकॉर्ड क्यों नहीं हैं - प्रत्येक पृष्ठ को हटाए जाने के लिए एक। ऐसा इसलिए है क्योंकि पेज डीललोकेशन भी कुशलता से लॉग किया गया है - एक लॉग रिकॉर्ड के साथ पीएफएस पेज में लगातार 8 डेटा फ़ाइल पेजों के लिए पीएफएस पेज आवंटन परिवर्तन को दर्शाता है, प्रत्येक डेटा फ़ाइल पेज के लिए एक लॉग रिकॉर्ड के बजाय पीएफएस पेज में इसकी आवंटन स्थिति को दर्शाता है।

वर्तमान पुनर्प्राप्ति मॉडल के आधार पर पूर्ण या न्यूनतम लॉगिंग के नियमों का पालन करते हुए SQL सर्वर हमेशा जितना संभव हो उतना कम लेनदेन लॉग उत्पन्न करने का प्रयास करता है। यदि आप अनहुक-एंड-ट्रांसफर और डिफर्ड-ड्रॉप तंत्र द्वारा उत्पन्न वास्तविक लॉग रिकॉर्ड को देखना चाहते हैं, तो बस उपरोक्त fn_dblog कोड में COUNT (*) के लिए * स्थानापन्न करें और लेन-देन नाम के साथ लेनदेन की तलाश करें DeferredAllocUnitDrop::Process

भविष्य की पोस्ट में मैं उन इंटर्नल पर चर्चा करूंगा जो SQL सर्वर स्टोरेज इंजन के प्रदर्शन पहलुओं के आसपास अन्य लगातार मिथकों को रेखांकित करते हैं।


  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. निष्पादन योजना ऑपरेटर समय को समझना

  2. अपने संग्रहण सबसिस्टम का परीक्षण करने के लिए Microsoft DiskSpd का उपयोग करना

  3. समूहीकृत कुल पुशडाउन

  4. वृद्धिशील आँकड़ों के साथ विभाजन रखरखाव में सुधार

  5. isql . में कमांड इतिहास