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

नल मूल्यों के साथ चयन में नहीं

पहले थोड़ा सिद्धांत:नल (एसक्यूएल)

उपरोक्त लिंक से हमारे लिए सबसे महत्वपूर्ण भाग:

<ब्लॉकक्वॉट>

NULL और तीन-मूल्यवान तर्क (3VL) के साथ तुलना

चूंकि नल किसी भी डेटा डोमेन का सदस्य नहीं है, इसलिए इसे "मान" नहीं माना जाता है, बल्कि एक मार्कर (या प्लेसहोल्डर) माना जाता है जो मूल्य के अभाव को दर्शाता है। इस वजह से, नल के साथ तुलना का परिणाम कभी भी सही या गलत नहीं हो सकता है, लेकिन हमेशा तीसरे तार्किक परिणाम में, अज्ञात होता है। [8] नीचे दिए गए व्यंजक का तार्किक परिणाम, जो मान 10 की तुलना शून्य से करता है, अज्ञात है:

SELECT 10 = NULL       -- Results in Unknown

ताकि दोनों तुलनाएं:x = NULL और x <> NULL NULL (अज्ञात) का मूल्यांकन करता है।

<ब्लॉकक्वॉट>

SQL तीन तार्किक परिणामों को लागू करता है, इसलिए SQL कार्यान्वयन को एक विशेष तीन-मूल्यवान तर्क (3VL) के लिए प्रदान करना चाहिए। एसक्यूएल तीन-मूल्यवान तर्क को नियंत्रित करने वाले नियम नीचे दी गई तालिकाओं में दिखाए गए हैं (पी और क्यू तार्किक राज्यों का प्रतिनिधित्व करते हैं) जो उनके निहितार्थ की परिभाषा में भिन्न है, हालांकि SQL ऐसे किसी भी ऑपरेशन को परिभाषित नहीं करता है)।

+---------+-------------+-------------+-------------+-----------+--------+
|    p    |        q    |     p OR q  |     p AND q |    p = q  |p != q  |
+---------+-------------+-------------+-------------+-----------+--------+
| True    |     True    |     True    |     True    |   True    | False  |
| True    |     False   |     True    |     False   |   False   | True   |
| True    |     Unknown |     True    |     Unknown |   Unknown | Unknown|
| False   |     True    |     True    |     False   |   False   | True   |
| False   |     False   |     False   |     False   |   True    | False  |
| False   |     Unknown |     Unknown |     False   |   Unknown | Unknown|
| Unknown |     True    |     True    |     Unknown |   Unknown | Unknown|
| Unknown |     False   |     Unknown |     False   |   Unknown | Unknown|
| Unknown |     Unknown |     Unknown |     Unknown |   Unknown | Unknown|
+---------+-------------+-------------+-------------+-----------+--------+
<ब्लॉकक्वॉट>

WHERE क्लॉज में अज्ञात का प्रभाव

SQL तीन-मूल्यवान तर्क DML कथनों और प्रश्नों की तुलना में डेटा हेरफेर भाषा (DML) में सामने आया है। WHERE क्लॉज के कारण DML स्टेटमेंट केवल उन्हीं पंक्तियों पर कार्य करता है, जिसके लिए विधेय का मूल्यांकन True होता है।

तो संक्षेप में:WHERE क्लॉज NULL को FALSE मानता है

अब कृपया एक सरल मामले पर विचार करें:

SELECT * FROM T1;

|      X |
|--------|
|      1 |
| (null) |

और एक प्रश्न:

SELECT * FROM t1 WHERE x IN (1, NULL);

ऊपर दी गई क्वेरी इसके लिए एक शॉर्टलैंड है:

SELECT * FROM t1 
WHERE x = 1
  OR  x = NULL

तालिका से दूसरी पंक्ति के लिए t ( x =NULL) यह स्थिति कुछ इस तरह दिखती है:

WHERE NULL = 1
   OR NULL = NULL

तो पंक्ति के लिए यह शर्त x=NULL NULL का मूल्यांकन करता है क्योंकि NULL=1 NULL है, NULL=NULL NULL है, और NULL OR NULL NULL भी है (कृपया ऊपर दी गई तालिका 3VL देखें)।

अब अधिक जिज्ञासु मामले पर विचार करें:

SELECT * FROM t1 WHERE x NOT IN (1, NULL);

यह खंड x NOT IN (1, NULL) NOT ( x IN (1, NULL) )
तो यह इसके बराबर भी है:

NOT (
  x = 1
  OR
  x = NULL
)

और डी मॉर्गन के नियमों के अनुसार यह इसके बराबर है:

NOT ( x = 1 ) AND NOT ( x = NULL )

और (यदि हम NOT x = y . को प्रतिस्थापित करते हैं x <> y . के साथ ) यह इसके बराबर भी है:

 x <> 1 AND x <> NULL

कृपया अंतिम स्थिति को ध्यान से देखें:

WHERE 
x <> 1 AND x <> NULL

हम x <> NULL . से अधिक जानते हैं हमेशा NULL का मूल्यांकन करता है। हम उपरोक्त 3VL तालिका से यह भी जानते हैं कि true AND NULL both दोनों NULL और false AND NULL है FALSE का मूल्यांकन करता है, इसलिए पूरी स्थिति का मूल्यांकन हमेशा FALSE या NULL में किया जाता है, लेकिन यह कभी भी TRUE का मूल्यांकन नहीं करता है।

इसलिए इस शर्त के साथ एक प्रश्न:

SELECT .....
WHERE x NOT IN ( NULL, whatever)

हमेशा खाली परिणाम देता है

और अब आपकी क्वेरी, जो उत्सुक भी है:

SELECT * FROM t1
WHERE (id, val) NOT IN (select id, val from data2);

जिसे (निरंतर मानों का उपयोग करके) फिर से लिखा जा सकता है:

SELECT * FROM t1
WHERE (id, val) NOT IN (
       (1, null),
       (2, 2 )
)

यह क्वेरी तथाकथित पंक्ति मान अभिव्यक्ति का उपयोग कर रही है

मूल रूप से इस तरह से पंक्ति मान एक्सप्रेसिन का उपयोग करने वाली एक शर्त

(a, b) = (x, y)

इसके बराबर है:

a = x AND b = y

इसलिए उपरोक्त क्वेरी को इसमें फिर से लिखा जा सकता है:

SELECT * FROM t1
WHERE NOT (
   id = 1 AND val = NULL
   OR
   id = 2 AND val = 2
)

डी मॉर्गन के नियमों के अनुसार यह समान है:

SELECT * FROM t1
WHERE 
   NOT ( id = 1 AND val = NULL )
   AND
   NOT ( id = 2 AND val = 2 )

और आगे:

SELECT * FROM t1
WHERE 
   ( id <> 1 OR val <> NULL )
   AND
   ( id <> 2 OR val <> 2 )

पहले भाग के बाद से ( id <> 1 OR val <> NULL ) शर्त का मूल्यांकन केवल उस स्थिति में होता है जब id <> 1 (कृपया ऊपर 3VL तालिका देखें), इस शर्त को सरल बनाया जा सकता है:

SELECT * FROM t1
WHERE 
   ( id <> 1 )
   AND
   ( id <> 2 OR val <> 2 )

और आगे (डी मॉर्गन के नियमों के अनुसार):

SELECT * FROM t1
WHERE 
   id <> 1 AND id <> 2
   OR
   id <> 1 AND  val <> 2

तो न तो (1,1) न ही (2,2) स्रोत से data1 इन शर्तों का पालन करें।



  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. सी #:ऑब्जेक्ट को डीबीएनयूएल से अन्य प्रकारों में नहीं डाला जा सकता है

  3. एंटिटी के साथ बिना प्राथमिक कुंजी वाले दृश्य का उपयोग करना

  4. Oracle में RPAD () फ़ंक्शन

  5. Oracle डेटाबेस में INTO क्लॉज के साथ तत्काल निष्पादन का उपयोग कैसे करें