यह पोस्ट नई जानकारी provides प्रदान करती है न्यूनतम रूप से लॉग किए गए बल्क लोड . के लिए पूर्व शर्त के बारे में INSERT...SELECT . का उपयोग करते समय अनुक्रमित तालिकाओं . में ।
इन मामलों को सक्षम करने वाली आंतरिक सुविधा को FastLoadContext . कहा जाता है . इसे दस्तावेज़ ट्रेस फ़्लैग 610 का उपयोग करके SQL सर्वर 2008 से 2014 तक सक्रिय किया जा सकता है। SQL सर्वर 2016 से आगे, FastLoadContext डिफ़ॉल्ट रूप से सक्षम है; ट्रेस ध्वज की आवश्यकता नहीं है।
बिना FastLoadContext . के , एकमात्र इंडेक्स इंसर्ट जो न्यूनतम लॉग किया जा सकता है क्या वे खाली . में हैं द्वितीयक अनुक्रमणिका के बिना संकुल अनुक्रमणिका, जैसा कि इस श्रृंखला के भाग दो में शामिल किया गया है। न्यूनतम लॉगिंग अनइंडेक्स्ड हीप टेबल के लिए शर्तों को पहले भाग में शामिल किया गया था।
अधिक पृष्ठभूमि के लिए, डेटा प्रदर्शन लोडिंग मार्गदर्शिका और टाइगर टीम देखें SQL सर्वर 2016 के लिए व्यवहार परिवर्तन पर नोट्स।
फास्ट लोड प्रसंग
एक त्वरित अनुस्मारक के रूप में, RowsetBulk सुविधा (भाग 1 और 2 में शामिल) न्यूनतम रूप से लॉग किए गए . को सक्षम करती है इसके लिए बल्क लोड:
- खाली और खाली नहीं ढेर के साथ टेबल:
- टेबल लॉकिंग; और
- कोई द्वितीयक अनुक्रमणिका नहीं।
- खाली संकुल तालिकाएं , साथ:
- टेबल लॉकिंग; और
- कोई द्वितीयक अनुक्रमणिका नहीं; और
DMLRequestSort=trueक्लस्टर इंडेक्स इंसर्ट . पर ऑपरेटर।
FastLoadContext कोड पथ न्यूनतम लॉग किए गए . के लिए समर्थन जोड़ता है और समवर्ती बल्क लोड चालू:
- खाली और गैर-रिक्त क्लस्टर किए गए बी-ट्री इंडेक्स।
- खाली और गैर-रिक्त गैर-संकुल एक समर्पित . द्वारा अनुरक्षित b-पेड़ अनुक्रमणिका इंडेक्स इंसर्ट योजना संचालक।
FastLoadContext DMLRequestSort=true की भी आवश्यकता है सभी मामलों में संबंधित योजना ऑपरेटर पर।
आपने RowsetBulk . के बीच एक ओवरलैप देखा होगा और FastLoadContext बिना सेकेंडरी इंडेक्स वाली खाली क्लस्टर टेबल के लिए। एक TABLOCK संकेत आवश्यक नहीं है FastLoadContext . के साथ , लेकिन यह अनुपस्थित होना आवश्यक नहीं है या। परिणामस्वरूप, TABLOCK . के साथ एक उपयुक्त इंसर्ट अभी भी न्यूनतम लॉगिंग के लिए योग्य हो सकता है FastLoadContext . के माध्यम से अगर यह विस्तृत RowsetBulk . में विफल रहता है परीक्षण।
FastLoadContext अक्षम किया जा सकता है SQL सर्वर 2016 पर प्रलेखित ट्रेस ध्वज 692 का उपयोग कर। डिबग चैनल विस्तारित घटना fastloadcontext_enabled FastLoadContext की निगरानी के लिए इस्तेमाल किया जा सकता है प्रति अनुक्रमणिका विभाजन (रोसेट) का उपयोग। यह इवेंट RowsetBulk . के लिए सक्रिय नहीं होता है भार।
मिश्रित लॉगिंग
एक ही INSERT...SELECT FastLoadContext . का उपयोग करके कथन पूरी तरह से लॉग कर सकते हैं न्यूनतम लॉगिंग . के दौरान कुछ पंक्तियां अन्य।
पंक्तियाँ डाली जाती हैं एक बार में एक इंडेक्स इंसर्ट . द्वारा ऑपरेटर और पूरी तरह से लॉग इन निम्नलिखित मामलों में:
- सभी पंक्तियों को पहले . में जोड़ा गया इंडेक्स पेज, अगर इंडेक्स खाली था ऑपरेशन की शुरुआत में।
- पंक्तियों को मौजूदा में जोड़ा गया अनुक्रमणिका पृष्ठ।
- पंक्तियां स्थानांतरित पृष्ठों के बीच एक पृष्ठ विभाजन द्वारा।
अन्यथा, आदेशित सम्मिलित स्ट्रीम से पंक्तियाँ एक बिल्कुल नए पृष्ठ . में जोड़ दी जाती हैं एक अनुकूलित, और न्यूनतम लॉग किए गए . का उपयोग करके कोड पथ। एक बार नए पृष्ठ पर जितनी संभव हो उतनी पंक्तियाँ लिखी जाती हैं, तो यह सीधे मौजूदा लक्ष्य सूचकांक संरचना से जुड़ जाती है।
नया जोड़ा गया पेज जरूरी नहीं . होगा पूर्ण हो (हालांकि स्पष्ट रूप से यह आदर्श मामला है) क्योंकि SQL सर्वर को सावधान रहना होगा कि नए पृष्ठ पर पंक्तियों को न जोड़ें जो तार्किक रूप से मौजूदा से संबंधित हैं सूचकांक पेज। नया पृष्ठ एक इकाई के रूप में सूचकांक में 'सिलाई' किया जाएगा, इसलिए हमारे पास नए पृष्ठ पर ऐसी कोई पंक्तियाँ नहीं हो सकतीं जो कहीं और से संबंधित हों। भीतर rows पंक्तियों को जोड़ते समय यह मुख्य रूप से एक समस्या है इंडेक्स की मौजूदा कुंजी श्रेणी, प्रारंभ से पहले या मौजूदा अनुक्रमणिका कुंजी श्रेणी के अंत के बाद के बजाय।
यह अभी भी संभव है अंदर . में नए पृष्ठ जोड़ने के लिए मौजूदा अनुक्रमणिका कुंजी श्रेणी, लेकिन नई पंक्तियों को पूर्ववर्ती पर उच्चतम कुंजी से ऊपर क्रमबद्ध करना चाहिए मौजूदा अनुक्रमणिका पृष्ठ और निम्नलिखित . पर निम्नतम कुंजी से नीचे क्रमबद्ध करें मौजूदा सूचकांक पृष्ठ। न्यूनतम लॉगिंग achieving प्राप्त करने के सर्वोत्तम अवसर के लिए इन परिस्थितियों में, सुनिश्चित करें कि सम्मिलित पंक्तियाँ जहाँ तक संभव हो मौजूदा पंक्तियों के साथ ओवरलैप न हों।
DMLRequestSort शर्तें
याद रखें कि FastLoadContext केवल तभी सक्रिय किया जा सकता है जब DMLRequestSort सत्य . पर सेट है संबंधित इंडेक्स इंसर्ट . के लिए निष्पादन योजना में ऑपरेटर।
दो मुख्य कोड पथ हैं जो DMLRequestSort . सेट कर सकते हैं करने के लिए सच सूचकांक डालने के लिए। कोई भी रास्ता लौट रहा है सच पर्याप्त है।
1. FOptimizeInsert
sqllang!CUpdUtil::FOptimizeInsert कोड की आवश्यकता है:
- 250 से अधिक पंक्तियां अनुमानित डाला जाना; और
- 2 से अधिक पृष्ठ अनुमानित डेटा आकार डालें; और
- लक्ष्य अनुक्रमणिका 3 से कम पत्ती वाले पृष्ठ . होने चाहिए ।
ये शर्तें RowsetBulk . जैसी ही हैं एक खाली क्लस्टर इंडेक्स पर, दो से अधिक इंडेक्स लीफ-लेवल पेजों के लिए अतिरिक्त आवश्यकता नहीं है। ध्यान से नोट करें कि यह मौजूदा अनुक्रमणिका . के आकार को दर्शाता है डालने से पहले, नहीं जोड़े जाने वाले डेटा का अनुमानित आकार।
नीचे दी गई स्क्रिप्ट इस श्रृंखला में पहले के हिस्सों में इस्तेमाल किए गए डेमो का एक संशोधन है। यह न्यूनतम लॉगिंग . दिखाता है जब तीन से कम अनुक्रमणिका पृष्ठ पहले . भर जाते हैं परीक्षण INSERT...SELECT रन। परीक्षण तालिका स्कीमा ऐसा है कि डेटाबेस के लिए पंक्ति संस्करण बंद होने पर 130 पंक्तियाँ एकल 8KB पृष्ठ पर फ़िट हो सकती हैं। पहले TOP . में गुणक मौजूदा इंडेक्स पेजों की संख्या पहले . निर्धारित करने के लिए क्लॉज को बदला जा सकता है परीक्षण INSERT...SELECT निष्पादित किया जाता है:
IF OBJECT_ID(N'dbo.Test', N'U') IS NOT NULL
BEGIN
DROP TABLE dbo.Test;
END;
GO
CREATE TABLE dbo.Test
(
id integer NOT NULL IDENTITY
CONSTRAINT [PK dbo.Test (id)]
PRIMARY KEY,
c1 integer NOT NULL,
padding char(45) NOT NULL
DEFAULT ''
);
GO
-- 130 rows per page for this table
-- structure with row versioning off
INSERT dbo.Test
(c1)
SELECT TOP (3 * 130) -- Change the 3 here
CHECKSUM(NEWID())
FROM master.dbo.spt_values AS SV;
GO
-- Show physical index statistics
-- to confirm the number of pages
SELECT
DDIPS.index_type_desc,
DDIPS.alloc_unit_type_desc,
DDIPS.page_count,
DDIPS.record_count,
DDIPS.avg_record_size_in_bytes
FROM sys.dm_db_index_physical_stats
(
DB_ID(),
OBJECT_ID(N'dbo.Test', N'U'),
1, -- Index ID
NULL, -- Partition ID
'DETAILED'
) AS DDIPS
WHERE
DDIPS.index_level = 0; -- leaf level only
GO
-- Clear the plan cache
DBCC FREEPROCCACHE;
GO
-- Clear the log
CHECKPOINT;
GO
-- Main test
INSERT dbo.Test
(c1)
SELECT TOP (269)
CHECKSUM(NEWID())
FROM master.dbo.spt_values AS SV;
GO
-- Show log entries
SELECT
FD.Operation,
FD.Context,
FD.[Log Record Length],
FD.[Log Reserve],
FD.AllocUnitName,
FD.[Transaction Name],
FD.[Lock Information],
FD.[Description]
FROM sys.fn_dblog(NULL, NULL) AS FD;
GO
-- Count the number of fully-logged rows
SELECT
[Fully Logged Rows] = COUNT_BIG(*)
FROM sys.fn_dblog(NULL, NULL) AS FD
WHERE
FD.Operation = N'LOP_INSERT_ROWS'
AND FD.Context = N'LCX_CLUSTERED'
AND FD.AllocUnitName = N'dbo.Test.PK dbo.Test (id)';
GO जब संकुल अनुक्रमणिका 3 पृष्ठों के साथ पहले से लोड हो जाती है , परीक्षण प्रविष्टि पूरी तरह से लॉग है (संक्षिप्तता के लिए छोड़े गए लेनदेन लॉग विवरण रिकॉर्ड):

जब तालिका केवल 1 या 2 पृष्ठों के साथ पहले से लोड हो जाती है , परीक्षण सम्मिलन न्यूनतम लॉग किया गया . है :


जब तालिका प्रीलोडेड नहीं है किसी भी पृष्ठ के साथ, परीक्षण भाग दो से खाली क्लस्टर तालिका डेमो चलाने के बराबर है, लेकिन बिना TABLOCK संकेत:

पहली 130 पंक्तियाँ पूरी तरह से लॉग की गई हैं . ऐसा इसलिए है क्योंकि हमारे शुरू होने से पहले सूचकांक खाली था, और 130 पंक्तियाँ पहले पृष्ठ पर फिट होती हैं। याद रखें, पहला पेज हमेशा पूरी तरह लॉग होता है जब FastLoadContext का उपयोग किया जाता है और सूचकांक पहले से खाली था। शेष 139 पंक्तियों को न्यूनतम लॉगिंग . के साथ सम्मिलित किया गया है ।
अगर कोई TABLOCK इंसर्ट में हिंट जोड़ा जाता है, सभी पेज कम से कम लॉग किए गए हैं (पहले वाले सहित) क्योंकि खाली क्लस्टर इंडेक्स लोड अब RowsetBulk के लिए योग्य है तंत्र (Sch-M . लेने की कीमत पर) ताला)।
2. FDemandRowsSortedForPerformance
अगर FOptimizeInsert परीक्षण विफल, DMLRequestSort अभी भी सत्य . पर सेट किया जा सकता है sqllang!CUpdUtil::FDemandRowsSortedForPerformance में परीक्षणों के दूसरे सेट द्वारा कोड। ये शर्तें थोड़ी अधिक जटिल हैं, इसलिए कुछ मापदंडों को परिभाषित करना उपयोगी होगा:
P– मौजूदा पत्ती-स्तरीय पृष्ठों . की संख्या लक्षित अनुक्रमणिका . में ।I- अनुमानित सम्मिलित करने के लिए पंक्तियों की संख्या।R=P/I(प्रत्येक सम्मिलित पंक्ति के लिए लक्षित पृष्ठ)।T- लक्ष्य विभाजन की संख्या (विभाजित के लिए 1)।
DMLRequestSort . का मान निर्धारित करने का तर्क तब है:
- यदि
P <= 16वापसी झूठी , अन्यथा :- यदि
R < 8:- यदि
P > 524वापसी सत्य , अन्यथा गलत ।
- यदि
- यदि
R >= 8:- अगर
T > 1औरI > 250वापसी सत्य , अन्यथा गलत ।
- अगर
- यदि
उपरोक्त परीक्षणों का मूल्यांकन योजना संकलन के दौरान क्वेरी प्रोसेसर द्वारा किया जाता है। एक अंतिम शर्त है स्टोरेज इंजन कोड द्वारा मूल्यांकन किया गया (IndexDataSetSession::WakeUpInternal ) निष्पादन के समय:
DMLRequestSortवर्तमान में सत्य है; औरI >= 100।
हम इस सारे तर्क को आगे प्रबंधनीय टुकड़ों में तोड़ देंगे।
16 से अधिक मौजूदा लक्ष्य पृष्ठ
पहला परीक्षण P <= 16 इसका मतलब है कि 17 से कम मौजूदा लीफ पेज वाले इंडेक्स FastLoadContext . के लिए योग्य नहीं होंगे इस कोड पथ के माध्यम से। इस बिंदु पर बिल्कुल स्पष्ट होने के लिए, P पहले . लक्ष्य अनुक्रमणिका में लीफ़-स्तरीय पृष्ठों की संख्या है INSERT...SELECT निष्पादित किया जाता है।
तर्क के इस भाग को प्रदर्शित करने के लिए, हम परीक्षण संकुल तालिका को 16 पृष्ठों . के साथ पहले से लोड करेंगे आंकड़े का। इसके दो महत्वपूर्ण प्रभाव हैं (याद रखें कि दोनों कोड पथ गलत लौटना चाहिए एक गलत . के साथ समाप्त करने के लिए DMLRequestSort . के लिए मान ):
- यह सुनिश्चित करता है कि पिछला
FOptimizeInsertपरीक्षण विफल , क्योंकि तीसरी शर्त पूरी नहीं हुई है (P < 3)। - द
P <= 16FDemandRowsSortedForPerformance. में स्थिति नहीं भी होगा मिलना चाहिए।
इसलिए हम उम्मीद करते हैं FastLoadContext सक्षम नहीं होना है। संशोधित डेमो स्क्रिप्ट है:
IF OBJECT_ID(N'dbo.Test', N'U') IS NOT NULL
BEGIN
DROP TABLE dbo.Test;
END;
GO
CREATE TABLE dbo.Test
(
id integer NOT NULL IDENTITY
CONSTRAINT [PK dbo.Test (id)]
PRIMARY KEY,
c1 integer NOT NULL,
padding char(45) NOT NULL
DEFAULT ''
);
GO
-- 130 rows per page for this table
-- structure with row versioning off
INSERT dbo.Test
(c1)
SELECT TOP (16 * 130) -- 16 pages
CHECKSUM(NEWID())
FROM master.dbo.spt_values AS SV;
GO
-- Show physical index statistics
-- to confirm the number of pages
SELECT
DDIPS.index_type_desc,
DDIPS.alloc_unit_type_desc,
DDIPS.page_count,
DDIPS.record_count,
DDIPS.avg_record_size_in_bytes
FROM sys.dm_db_index_physical_stats
(
DB_ID(),
OBJECT_ID(N'dbo.Test', N'U'),
1, -- Index ID
NULL, -- Partition ID
'DETAILED'
) AS DDIPS
WHERE
DDIPS.index_level = 0; -- leaf level only
GO
-- Clear the plan cache
DBCC FREEPROCCACHE;
GO
-- Clear the log
CHECKPOINT;
GO
-- Main test
INSERT dbo.Test
(c1)
SELECT TOP (269)
CHECKSUM(NEWID())
FROM master.dbo.spt_values AS SV1
CROSS JOIN master.dbo.spt_values AS SV2;
GO
-- Show log entries
SELECT
FD.Operation,
FD.Context,
FD.[Log Record Length],
FD.[Log Reserve],
FD.AllocUnitName,
FD.[Transaction Name],
FD.[Lock Information],
FD.[Description]
FROM sys.fn_dblog(NULL, NULL) AS FD;
GO
-- Count the number of fully-logged rows
SELECT
[Fully Logged Rows] = COUNT_BIG(*)
FROM sys.fn_dblog(NULL, NULL) AS FD
WHERE
FD.Operation = N'LOP_INSERT_ROWS'
AND FD.Context = N'LCX_CLUSTERED'
AND FD.AllocUnitName = N'dbo.Test.PK dbo.Test (id)'; सभी 269 पंक्तियाँ पूरी तरह से लॉग हैं भविष्यवाणी के अनुसार:

ध्यान दें कि हम सम्मिलित करने के लिए नई पंक्तियों की संख्या कितनी भी अधिक क्यों न सेट करें, ऊपर दी गई स्क्रिप्ट कभी नहीं होगी न्यूनतम लॉगिंग उत्पन्न करें P <= 16 . के कारण परीक्षण (और P < 3 FOptimizeInsert . में परीक्षण करें )।
यदि आप बड़ी संख्या में पंक्तियों के साथ स्वयं डेमो चलाना चुनते हैं, तो उस अनुभाग पर टिप्पणी करें जो व्यक्तिगत लेनदेन लॉग रिकॉर्ड दिखाता है, अन्यथा आप बहुत लंबे समय तक प्रतीक्षा करेंगे, और SSMS क्रैश हो सकता है। (निष्पक्ष होने के लिए, यह वैसे भी ऐसा कर सकता है, लेकिन जोखिम में क्यों जोड़ें।)
पृष्ठ प्रति सम्मिलित पंक्ति अनुपात
यदि 17 या अधिक . हैं तो मौजूदा इंडेक्स में लीफ पेज, पिछला P <= 16 परीक्षण विफल नहीं होगा। तर्क का अगला भाग मौजूदा पृष्ठों . के अनुपात से संबंधित है नई सम्मिलित पंक्तियों . के लिए . इसे न्यूनतम लॉगिंग . प्राप्त करने के लिए भी पास करना होगा . एक अनुस्मारक के रूप में, प्रासंगिक शर्तें हैं:
- अनुपात
R=P/I। - यदि
R < 8:- यदि
P > 524वापसी सत्य , अन्यथा गलत ।
- यदि
हमें कम से कम 100 पंक्तियों के लिए अंतिम भंडारण इंजन परीक्षण भी याद रखना चाहिए:
I >= 100।
उन शर्तों को थोड़ा सा पुनर्व्यवस्थित करना, सभी निम्नलिखित में से सत्य होना चाहिए:
P > 524(मौजूदा इंडेक्स पेज)I >= 100(अनुमानित सम्मिलित पंक्तियाँ)P / I < 8(अनुपातR)
उन तीन शर्तों को एक साथ पूरा करने के कई तरीके हैं। आइए P . के लिए न्यूनतम संभव मान चुनें (525) और I (100) एक R दे रहा है (525/100) का मान =5.25। यह संतुष्ट करता है (R < 8 परीक्षण), इसलिए हम उम्मीद करते हैं कि इस संयोजन का परिणाम न्यूनतम लॉगिंग . होगा :
IF OBJECT_ID(N'dbo.Test', N'U') IS NOT NULL
BEGIN
DROP TABLE dbo.Test;
END;
GO
CREATE TABLE dbo.Test
(
id integer NOT NULL IDENTITY
CONSTRAINT [PK dbo.Test (id)]
PRIMARY KEY,
c1 integer NOT NULL,
padding char(45) NOT NULL
DEFAULT ''
);
GO
-- 130 rows per page for this table
-- structure with row versioning off
INSERT dbo.Test
(c1)
SELECT TOP (525 * 130) -- 525 pages
CHECKSUM(NEWID())
FROM master.dbo.spt_values AS SV1
CROSS JOIN master.dbo.spt_values AS SV2;
GO
-- Show physical index statistics
-- to confirm the number of pages
SELECT
DDIPS.index_type_desc,
DDIPS.alloc_unit_type_desc,
DDIPS.page_count,
DDIPS.record_count,
DDIPS.avg_record_size_in_bytes
FROM sys.dm_db_index_physical_stats
(
DB_ID(),
OBJECT_ID(N'dbo.Test', N'U'),
1, -- Index ID
NULL, -- Partition ID
'DETAILED'
) AS DDIPS
WHERE
DDIPS.index_level = 0; -- leaf level only
GO
-- Clear the plan cache
DBCC FREEPROCCACHE;
GO
-- Clear the log
CHECKPOINT;
GO
-- Main test
INSERT dbo.Test
(c1)
SELECT TOP (100)
CHECKSUM(NEWID())
FROM master.dbo.spt_values AS SV1
CROSS JOIN master.dbo.spt_values AS SV2;
GO
-- Show log entries
SELECT
FD.Operation,
FD.Context,
FD.[Log Record Length],
FD.[Log Reserve],
FD.AllocUnitName,
FD.[Transaction Name],
FD.[Lock Information],
FD.[Description]
FROM sys.fn_dblog(NULL, NULL) AS FD;
GO
-- Count the number of fully-logged rows
SELECT
[Fully Logged Rows] = COUNT_BIG(*)
FROM sys.fn_dblog(NULL, NULL) AS FD
WHERE
FD.Operation = N'LOP_INSERT_ROWS'
AND FD.Context = N'LCX_CLUSTERED'
AND FD.AllocUnitName = N'dbo.Test.PK dbo.Test (id)';
100-पंक्ति INSERT...SELECT वास्तव में न्यूनतम लॉग किया गया है :

अनुमानित . को कम करना 99 में पंक्तियाँ डाली गईं (I >= 100 को तोड़ते हुए ), और/या मौजूदा अनुक्रमणिका पृष्ठों की संख्या को घटाकर 524 कर दें (P > 524 . को तोड़कर) ) परिणाम पूर्ण लॉगिंग . में होता है . हम ऐसे बदलाव भी कर सकते हैं जैसे R पूर्ण लॉगिंग produce उत्पन्न करने के लिए अब 8 से कम नहीं है . उदाहरण के लिए, P = 1000 setting सेट करना और I = 125 देता है R = 8 , निम्नलिखित परिणामों के साथ:

डाली गई 125 पंक्तियां पूरी तरह से लॉग थीं जैसा सोचा था। (यह प्रथम पृष्ठ पूर्ण लॉगिंग के कारण नहीं है, क्योंकि अनुक्रमणिका पहले से खाली नहीं थी।)
विभाजित अनुक्रमणिका के लिए पृष्ठ अनुपात
यदि पिछले सभी परीक्षण विफल हो जाते हैं, तो शेष एक परीक्षण के लिए R >= 8 . की आवश्यकता होती है और केवल . कर सकते हैं विभाजनों की संख्या (T .) होने पर संतुष्ट रहें ) 1 और . से बड़ा है 250 से अधिक अनुमानित . हैं सम्मिलित पंक्तियाँ (I ) याद करें:
- यदि
R >= 8:- अगर
T > 1औरI > 250वापसी सत्य , अन्यथा गलत ।
- अगर
एक सूक्ष्मता:विभाजित . के लिए अनुक्रमणिका, नियम जो कहता है कि सभी प्रथम-पृष्ठ पंक्तियाँ पूरी तरह से लॉग हैं (शुरुआत में खाली अनुक्रमणिका के लिए) प्रति विभाजन लागू होता है . 15,000 विभाजन वाले ऑब्जेक्ट के लिए, इसका मतलब है कि 15,000 पूरी तरह से लॉग किए गए 'पहले' पृष्ठ।
सारांश और अंतिम विचार
मुख्य भाग में वर्णित मूल्यांकन के सूत्र और क्रम डिबगर का उपयोग करके कोड निरीक्षण पर आधारित होते हैं। उन्हें एक ऐसे रूप में प्रस्तुत किया गया जो वास्तविक कोड में उपयोग किए गए समय और क्रम का बारीकी से प्रतिनिधित्व करता है।
न्यूनतम लॉगिंग के लिए व्यावहारिक आवश्यकताओं का अधिक संक्षिप्त सारांश तैयार करने के लिए, उन स्थितियों को थोड़ा सा पुन:व्यवस्थित और सरल बनाना संभव है INSERT...SELECT . का उपयोग करके बी-पेड़ में डालने पर . नीचे दिए गए परिष्कृत भाव निम्नलिखित तीन मापदंडों का उपयोग करते हैं:
P=मौजूदा . की संख्या इंडेक्स लीफ-लेवल पेज.I=अनुमानित सम्मिलित करने के लिए पंक्तियों की संख्या।S=अनुमानित 8KB पृष्ठों में डेटा आकार डालें।
रोसेट बल्क लोड
sqlmin!RowsetBulkका उपयोग करता है ।- एक खाली की आवश्यकता है
TABLOCK. के साथ संकुल अनुक्रमणिका लक्ष्य (या समकक्ष)। - आवश्यकता है
DMLRequestSort = trueक्लस्टर इंडेक्स इंसर्ट . पर ऑपरेटर। DMLRequestSortसेट हैtrueअगरI > 250औरS > 2।- सम्मिलित सभी पंक्तियां न्यूनतम रूप से लॉग की गई हैं ।
- एक
Sch-Mलॉक समवर्ती टेबल एक्सेस को रोकता है।
तेज़ लोड प्रसंग
sqlmin!FastLoadContextका उपयोग करता है ।- सक्षम करता हैन्यूनतम लॉग किया हुआ बी-ट्री इंडेक्स में सम्मिलित करता है:
- संकुलित या गैर संकुलित।
- टेबल लॉक के साथ या बिना।
- लक्ष्य अनुक्रमणिका खाली है या नहीं।
- आवश्यकता है
DMLRequestSort = trueसंबंधित इंडेक्स इंसर्ट . पर योजना संचालक। - ब्रांड के लिए लिखी गई केवल पंक्तियाँ नए पृष्ठ बल्क लोडेड हैं और न्यूनतम रूप से लॉग किए गए हैं ।
- पहला पृष्ठ पहले के खाली अनुक्रमणिका . का विभाजन हमेशा पूरी तरह से लॉग होता है ।
- पूर्ण न्यूनतम
I >= 100। - SQL सर्वर 2016 से पहले ट्रेस ध्वज 610 की आवश्यकता है।
- SQL सर्वर 2016 से डिफ़ॉल्ट रूप से उपलब्ध (ट्रेस ध्वज 692 अक्षम करता है)।
DMLRequestSort सेट है true के लिए:
- कोई भी अनुक्रमणिका (विभाजित या नहीं) यदि:
I > 250औरP < 3औरS > 2; याI >= 100औरP > 524औरP < I * 8
केवल विभाजित अनुक्रमणिका . के लिए (> 1 विभाजन के साथ), DMLRequestSort true . भी सेट किया गया है अगर:
I > 250औरP > 16औरP >= I * 8
उन FastLoadContext . से कुछ दिलचस्प मामले सामने आ रहे हैं शर्तें:
- सभी एक गैर-विभाजित . में सम्मिलित करता है 3 और 524 के बीच . के साथ अनुक्रमणिका (समावेशी) मौजूदा लीफ पेज पूरी तरह से लॉग हो जाएंगे जोड़े गए पंक्तियों की संख्या और कुल आकार की परवाह किए बिना। यह छोटे (लेकिन खाली नहीं) टेबल पर बड़े इंसर्ट को सबसे ज्यादा प्रभावित करेगा।
- सभी विभाजित . में सम्मिलित करता है 3 और 16 . के बीच अनुक्रमणिका मौजूदा पृष्ठ पूरी तरह से लॉग होंगे ।
- बड़ी प्रविष्टियां बड़े पैमाने पर गैर-विभाजित अनुक्रमणिका न्यूनतम रूप से लॉग नहीं हो सकतीं असमानता के कारण
P < I * 8. जबPबड़ा है, एक संगत रूप से बड़ा अनुमानित सम्मिलित पंक्तियों की संख्या (I) आवश्यक है। उदाहरण के लिए, 8 मिलियन पृष्ठों वाली अनुक्रमणिका न्यूनतम लॉगिंग . का समर्थन नहीं कर सकती है 1 मिलियन या उससे कम पंक्तियाँ सम्मिलित करते समय।
गैर-संकुल अनुक्रमणिका
डेमो में क्लस्टर किए गए इंडेक्स पर लागू समान विचार और गणना गैर-क्लस्टर किए गए पर लागू होते हैं बी-ट्री इंडेक्स भी, जब तक कि इंडेक्स एक समर्पित योजना ऑपरेटर द्वारा बनाए रखा जाता है (एक चौड़ा , या प्रति-सूचकांक योजना)। बेस टेबल ऑपरेटर द्वारा अनुरक्षित गैर-संकुल अनुक्रमणिका (उदा. संकुल अनुक्रमणिका सम्मिलित करें ) FastLoadContext . के लिए योग्य नहीं हैं ।
ध्यान दें कि प्रत्येक गैर-संकुल . के लिए सूत्र पैरामीटर का नए सिरे से मूल्यांकन करने की आवश्यकता है इंडेक्स ऑपरेटर — परिकलित पंक्ति आकार, मौजूदा इंडेक्स पेजों की संख्या, और कार्डिनैलिटी अनुमान।
सामान्य टिप्पणियां
कम कार्डिनैलिटी अनुमान . से सावधान रहें इंडेक्स इंसर्ट . पर ऑपरेटर, क्योंकि ये I . को प्रभावित करेंगे और S पैरामीटर। यदि कार्डिनैलिटी अनुमान त्रुटि के कारण थ्रेशोल्ड तक नहीं पहुंचा जाता है, तो इंसर्ट पूरी तरह से लॉग हो जाएगा ।
याद रखें कि DMLRequestSort योजना के साथ संचित है - पुन:उपयोग की गई योजना के प्रत्येक निष्पादन पर इसका मूल्यांकन नहीं किया जाता है। यह प्रसिद्ध पैरामीटर संवेदनशीलता समस्या (जिसे "पैरामीटर सूँघने" के रूप में भी जाना जाता है) का एक रूप पेश कर सकता है।
P . का मान (इंडेक्स लीफ पेज) रीफ्रेश नहीं है प्रत्येक कथन की शुरुआत में। वर्तमान कार्यान्वयन पूरे बैच . के मान को संचित करता है . इसके अप्रत्याशित दुष्प्रभाव हो सकते हैं। उदाहरण के लिए, एक TRUNCATE TABLE उसी बैच . में एक INSERT...SELECT . के रूप में रीसेट नहीं होगा P इस आलेख में वर्णित गणनाओं के लिए शून्य करने के लिए — वे पूर्व-छंटनी मूल्य का उपयोग करना जारी रखेंगे, और एक पुनर्संकलन मदद नहीं करेगा। अलग-अलग बैचों में बड़े बदलाव सबमिट करना एक समाधान है।
ट्रेस फ़्लैग्स
FDemandRowsSortedForPerformance . को बाध्य करना संभव है सत्य पर लौटने के लिए अदस्तावेजीकृत और असमर्थित . सेट करके ट्रेस फ्लैग 2332, जैसा कि मैंने डेटा बदलने वाले टी-एसक्यूएल प्रश्नों को ऑप्टिमाइज़ करने में लिखा था। जब TF 2332 सक्रिय होता है, तो प्रस्तुत करने के लिए अनुमानित पंक्तियों की संख्या अभी भी कम से कम 100 . होना चाहिए . TF 2332 न्यूनतम लॉगिंग . को प्रभावित करता है FastLoadContext . के लिए निर्णय केवल (यह DMLRequestSort के रूप में विभाजित ढेर के लिए प्रभावी है) संबंधित है, लेकिन इसका ढेर पर कोई प्रभाव नहीं पड़ता है, क्योंकि FastLoadContext केवल अनुक्रमणिका पर लागू होता है)।
एक विस्तृत/प्रति-सूचकांक गैर-संकुल अनुक्रमणिका रखरखाव के लिए योजना आकार को ट्रेस फ़्लैग 8790 (आधिकारिक रूप से प्रलेखित नहीं है, लेकिन एक नॉलेज बेस आलेख में उल्लेख किया गया है और साथ ही मेरे लेख में TF2332 के लिए लिंक किया गया है) का उपयोग करके रोस्टोर टेबल के लिए मजबूर किया जा सकता है।
संबंधित पठन
SQL सर्वर टीम से सुनील अग्रवाल द्वारा सभी:
- थोक आयात अनुकूलन क्या हैं?
- थोक आयात अनुकूलन (न्यूनतम लॉगिंग)
- SQL Server 2008 में न्यूनतम लॉगिंग परिवर्तन
- SQL Server 2008 (भाग-2) में न्यूनतम लॉगिंग परिवर्तन
- SQL Server 2008 (भाग-3) में न्यूनतम लॉगिंग परिवर्तन