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

में नहीं बनाम मौजूद नहीं है

मैं हमेशा NOT EXISTS . के लिए डिफ़ॉल्ट हूं ।

निष्पादन योजनाएं इस समय समान हो सकती हैं लेकिन यदि भविष्य में किसी भी कॉलम को NULL की अनुमति देने के लिए बदल दिया जाता है NOT IN संस्करण को और अधिक काम करने की आवश्यकता होगी (भले ही कोई NULL . न हो s वास्तव में डेटा में मौजूद हैं) और शब्दार्थ NOT IN अगर NULL ये हैं वर्तमान में वैसे भी होने की संभावना नहीं है जो आप चाहते हैं।

जब न तो Products.ProductID या [Order Details].ProductID अनुमति दें NULL NOT IN निम्नलिखित क्वेरी के समान व्यवहार किया जाएगा।

SELECT ProductID,
       ProductName
FROM   Products p
WHERE  NOT EXISTS (SELECT *
                   FROM   [Order Details] od
                   WHERE  p.ProductId = od.ProductId) 

सटीक योजना भिन्न हो सकती है लेकिन मेरे उदाहरण डेटा के लिए मुझे निम्नलिखित मिलते हैं।

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

/*Not valid syntax but better reflects the plan*/ 
SELECT p.ProductID,
       p.ProductName
FROM   Products p
       LEFT ANTI SEMI JOIN [Order Details] od
         ON p.ProductId = od.ProductId 

अगर [Order Details].ProductID है NULL -सक्षम क्वेरी तब बन जाती है

SELECT ProductID,
       ProductName
FROM   Products p
WHERE  NOT EXISTS (SELECT *
                   FROM   [Order Details] od
                   WHERE  p.ProductId = od.ProductId)
       AND NOT EXISTS (SELECT *
                       FROM   [Order Details]
                       WHERE  ProductId IS NULL) 

इसका कारण यह है कि सही शब्दार्थ यदि [Order Details] इसमें कोई भी NULL शामिल है ProductId s कोई परिणाम नहीं लौटाना है। इसे सत्यापित करने के लिए अतिरिक्त एंटी सेमी जॉइन और रो काउंट स्पूल देखें जो कि योजना में जोड़ा गया है।

अगर Products.ProductID NULL . में भी बदल दिया जाता है -सक्षम क्वेरी तब बन जाती है

SELECT ProductID,
       ProductName
FROM   Products p
WHERE  NOT EXISTS (SELECT *
                   FROM   [Order Details] od
                   WHERE  p.ProductId = od.ProductId)
       AND NOT EXISTS (SELECT *
                       FROM   [Order Details]
                       WHERE  ProductId IS NULL)
       AND NOT EXISTS (SELECT *
                       FROM   (SELECT TOP 1 *
                               FROM   [Order Details]) S
                       WHERE  p.ProductID IS NULL) 

इसका कारण यह है कि एक NULL Products.ProductId परिणामों में नहीं लौटाया जाना चाहिए सिवाय अगर NOT IN उप क्वेरी को कोई परिणाम नहीं लौटाना था (यानी [Order Details] टेबल खाली है)। किस मामले में चाहिए। मेरे नमूना डेटा की योजना में इसे नीचे के रूप में एक और एंटी सेमी जॉइन जोड़कर लागू किया गया है।

इसका प्रभाव बकले द्वारा पहले से लिंक किए गए ब्लॉग पोस्ट में दिखाया गया है। उदाहरण में तार्किक पढ़ने की संख्या लगभग 400 से 500,000 तक बढ़ जाती है।

इसके अतिरिक्त तथ्य यह है कि एक एकल NULL पंक्ति संख्या को शून्य तक कम कर सकता है जिससे कार्डिनैलिटी का अनुमान बहुत मुश्किल हो जाता है। यदि SQL सर्वर मानता है कि ऐसा होगा लेकिन वास्तव में कोई NULL नहीं था डेटा में पंक्तियाँ शेष निष्पादन योजना भयावह रूप से बदतर हो सकती हैं, यदि यह एक बड़ी क्वेरी का सिर्फ एक हिस्सा है, जिसमें अनुपयुक्त नेस्टेड लूप उदाहरण के लिए एक महंगे उप-वृक्ष के बार-बार निष्पादन का कारण बनते हैं।

NOT IN . के लिए यह एकमात्र संभावित निष्पादन योजना नहीं है एक NULL . पर -सक्षम कॉलम हालांकि। यह आलेख AdventureWorks2008 . के विरुद्ध एक प्रश्न के लिए दूसरा आलेख दिखाता है डेटाबेस।

NOT IN के लिए एक पर NOT NULL कॉलम या NOT EXISTS एक अशक्त या अशक्त स्तंभ के सामने यह निम्नलिखित योजना देता है।

जब कॉलम NULL में बदल जाता है -सक्षम NOT IN योजना अब इस तरह दिखती है

यह योजना में एक अतिरिक्त इनर जॉइन ऑपरेटर जोड़ता है। इस उपकरण को यहां समझाया गया है। पिछले एकल सहसंबद्ध अनुक्रमणिका को Sales.SalesOrderDetail.ProductID = <correlated_product_id> पर परिवर्तित करने के लिए यह सब कुछ है प्रति बाहरी पंक्ति में दो मांगता है। अतिरिक्त एक WHERE Sales.SalesOrderDetail.ProductID IS NULL पर है .

जैसा कि यह एक एंटी सेमी जॉइन के तहत है, अगर वह कोई पंक्ति लौटाता है तो दूसरी तलाश नहीं होगी। हालांकि अगर Sales.SalesOrderDetail इसमें कोई NULL नहीं है ProductId s यह आवश्यक खोज कार्यों की संख्या को दोगुना कर देगा।



  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. SQL सर्वर में N पंक्तियों का चयन करना

  3. SQL सर्वर डेटाबेस को Azure SQL डेटाबेस में माइग्रेट करने के तरीके

  4. मैं अपने SQL सर्वर एजेंट जॉब में एक चरण कैसे बनाऊं जो मेरा SSIS पैकेज चलाएगा?

  5. 'केवल पढ़ने के लिए' डेटाबेस को छोटा नहीं कर सकता | ऑलवेजऑन अवेलेबिलिटी ग्रुप का उपयोग करते हुए ट्रांजेक्शन लॉग को सिकोड़ें