CASE अभिव्यक्ति टी-एसक्यूएल में मेरी पसंदीदा संरचनाओं में से एक है। यह काफी लचीला है, और कभी-कभी उस क्रम को नियंत्रित करने का एकमात्र तरीका है जिसमें SQL सर्वर विधेय का मूल्यांकन करेगा।
हालांकि, इसे अक्सर गलत समझा जाता है।
टी-एसक्यूएल केस एक्सप्रेशन क्या है?
टी-एसक्यूएल में, CASE एक व्यंजक है जो एक या अधिक संभावित व्यंजकों का मूल्यांकन करता है और पहला उपयुक्त व्यंजक देता है। यहां अभिव्यक्ति शब्द थोड़ा अतिभारित हो सकता है, लेकिन मूल रूप से यह कुछ भी है जिसे एक एकल, स्केलर मान के रूप में मूल्यांकन किया जा सकता है, जैसे कि एक चर, एक कॉलम, एक स्ट्रिंग अक्षर, या यहां तक कि एक अंतर्निर्मित या स्केलर फ़ंक्शन का आउटपुट ।
T-SQL में CASE के दो रूप हैं:
- सरल केस एक्सप्रेशन - जब आपको केवल समानता का मूल्यांकन करने की आवश्यकता हो:
CASE <input> WHEN <eval> THEN <return> … [ELSE <return>] END - केस एक्सप्रेशन खोजा गया - जब आपको अधिक जटिल अभिव्यक्तियों का मूल्यांकन करने की आवश्यकता हो, जैसे कि असमानता, LIKE, या IS NOT NULL:
CASE WHEN <input_bool> THEN <return> … [ELSE <return>] END
रिटर्न एक्सप्रेशन हमेशा एक ही मान होता है, और आउटपुट डेटा प्रकार डेटा प्रकार वरीयता द्वारा निर्धारित किया जाता है।
जैसा कि मैंने कहा, CASE अभिव्यक्ति को अक्सर गलत समझा जाता है; यहां कुछ उदाहरण दिए गए हैं:
CASE एक एक्सप्रेशन है, स्टेटमेंट नहीं
संभवतः अधिकांश लोगों के लिए महत्वपूर्ण नहीं है, और शायद यह सिर्फ मेरा पांडित्य पक्ष है, लेकिन बहुत से लोग इसे CASE कहते हैं बयान - Microsoft सहित, जिसके दस्तावेज़ में कथन का उपयोग किया गया है और अभिव्यक्ति समय-समय पर अदला-बदली। मुझे यह थोड़ा कष्टप्रद लगता है (जैसे पंक्ति/रिकॉर्ड और कॉलम/फ़ील्ड ) और, जबकि यह ज्यादातर शब्दार्थ है, लेकिन एक अभिव्यक्ति और एक बयान के बीच एक महत्वपूर्ण अंतर है:एक अभिव्यक्ति एक परिणाम देता है। जब लोग CASE . के बारे में सोचते हैं एक कथन . के रूप में , यह इस तरह कोड को छोटा करने के प्रयोगों की ओर ले जाता है:
SELECT CASE [status]
WHEN 'A' THEN
StatusLabel = 'Authorized',
LastEvent = AuthorizedTime
WHEN 'C' THEN
StatusLabel = 'Completed',
LastEvent = CompletedTime
END
FROM dbo.some_table; या यह:
SELECT CASE WHEN @foo = 1 THEN
(SELECT foo, bar FROM dbo.fizzbuzz)
ELSE
(SELECT blat, mort FROM dbo.splunge)
END;
इस प्रकार का प्रवाह नियंत्रण तर्क CASE . के साथ संभव हो सकता है बयान अन्य भाषाओं में (जैसे VBScript), लेकिन Transact-SQL के CASE . में नहीं अभिव्यक्ति . CASE का उपयोग करने के लिए उसी क्वेरी लॉजिक में, आपको CASE . का उपयोग करना होगा अभिव्यक्ति प्रत्येक आउटपुट कॉलम के लिए:
SELECT
StatusLabel = CASE [status]
WHEN 'A' THEN 'Authorized'
WHEN 'C' THEN 'Completed' END,
LastEvent = CASE [status]
WHEN 'A' THEN AuthorizedTime
WHEN 'C' THEN CompletedTime END
FROM dbo.some_table; CASE हमेशा शॉर्ट सर्किट नहीं करेगा
आधिकारिक दस्तावेज में एक बार यह निहित किया गया था कि पूरी अभिव्यक्ति शॉर्ट-सर्किट होगी, जिसका अर्थ है कि यह बाएं से दाएं अभिव्यक्ति का मूल्यांकन करेगा, और मैच हिट होने पर मूल्यांकन करना बंद कर देगा:
CASE स्टेटमेंट [sic!] अपनी शर्तों का क्रमिक रूप से मूल्यांकन करता है और पहली शर्त के साथ रुकता है जिसकी स्थिति संतुष्ट होती है।हालाँकि, यह हमेशा सच नहीं होता है। और इसके श्रेय के लिए, एक अधिक वर्तमान संस्करण में, पृष्ठ ने एक ऐसे परिदृश्य की व्याख्या करने का प्रयास किया जहां इसकी गारंटी नहीं है। लेकिन इसे केवल कहानी का हिस्सा मिलता है:
कुछ स्थितियों में, CASE स्टेटमेंट [sic!] के इनपुट के रूप में एक्सप्रेशन के परिणाम प्राप्त करने से पहले एक एक्सप्रेशन का मूल्यांकन किया जाता है। इन भावों के मूल्यांकन में त्रुटियाँ संभव हैं। जब किसी CASE कथन [sic!] के तर्कों में प्रकट होने वाले समग्र भावों का मूल्यांकन पहले किया जाता है, फिर CASE कथन [sic!] को प्रदान किया जाता है। उदाहरण के लिए, MAX एग्रीगेट का मान उत्पन्न करते समय निम्न क्वेरी शून्य त्रुटि से विभाजित करती है। यह CASE व्यंजक का मूल्यांकन करने से पहले होता है।शून्य उदाहरण से विभाजित करना बहुत आसान है, और मैंने इसे dba.stackexchange.com पर इस उत्तर में प्रदर्शित किया:
DECLARE @i INT = 1; SELECT CASE WHEN @i = 1 THEN 1 ELSE MIN(1/0) END;
परिणाम:
संदेश 8134, स्तर 16, राज्य 1शून्य त्रुटि से विभाजित करें।
छोटे-छोटे उपाय हैं (जैसे ELSE (SELECT MIN(1/0)) END ), लेकिन यह कई लोगों के लिए एक वास्तविक आश्चर्य के रूप में आता है जिन्होंने बुक्स ऑनलाइन से उपरोक्त वाक्यों को याद नहीं किया है। इट्ज़िक बेन-गण (@ItzikBenGan) द्वारा एक निजी ई-मेल वितरण सूची पर बातचीत में मुझे पहली बार इस विशिष्ट परिदृश्य से अवगत कराया गया था, जिसे शुरू में Jaime Lafargue द्वारा अधिसूचित किया गया था। मैंने Connect #690017 में बग की सूचना दी:CASE / COALESCE हमेशा पाठ्य क्रम में मूल्यांकन नहीं करेगा; इसे तेजी से "डिजाइन द्वारा" के रूप में बंद कर दिया गया था। पॉल व्हाइट (ब्लॉग | @SQL_Kiwi) ने बाद में Connect #691535 :Aggregates Don't Follow the Semantics Of CASE दायर किया, और इसे "फिक्स्ड" के रूप में बंद कर दिया गया। इस मामले में सुधार, पुस्तकें ऑनलाइन लेख में स्पष्टीकरण था; अर्थात्, वह स्निपेट जिसे मैंने ऊपर कॉपी किया था।
यह व्यवहार खुद को कुछ अन्य, कम स्पष्ट परिदृश्यों में भी उत्पन्न कर सकता है। उदाहरण के लिए, Connect #780132 :FREETEXT() CASE स्टेटमेंट में मूल्यांकन के आदेश का सम्मान नहीं करता है (कोई एग्रीगेट शामिल नहीं है) यह दर्शाता है कि, ठीक है, CASE कुछ पूर्ण-पाठ कार्यों का उपयोग करते समय मूल्यांकन आदेश बाएं से दाएं होने की गारंटी नहीं है। उस आइटम पर, पॉल व्हाइट ने टिप्पणी की कि उन्होंने नए LAG() . का उपयोग करके भी कुछ ऐसा ही देखा है SQL सर्वर 2012 में पेश किया गया फ़ंक्शन। मेरे पास रेप्रो आसान नहीं है, लेकिन मैं उस पर विश्वास करता हूं, और मुझे नहीं लगता कि हमने सभी किनारे के मामलों का पता लगाया है जहां यह हो सकता है।
इसलिए, जब पूर्ण-पाठ खोज जैसी समग्र या गैर-देशी सेवाएं शामिल हों, तो कृपया CASE में शॉर्ट सर्किटिंग के बारे में कोई धारणा न बनाएं। अभिव्यक्ति।
RAND() का मूल्यांकन एक से अधिक बार किया जा सकता है
मैंने अक्सर लोगों को सरल . लिखते हुए देखा है CASE अभिव्यक्ति, इस तरह:
SELECT CASE @variable WHEN 1 THEN 'foo' WHEN 2 THEN 'bar' END
यह समझना महत्वपूर्ण है कि इसे खोज . के रूप में निष्पादित किया जाएगा CASE अभिव्यक्ति, इस तरह:
SELECT CASE WHEN @variable = 1 THEN 'foo' WHEN @variable = 2 THEN 'bar' END
यह समझना महत्वपूर्ण है कि मूल्यांकन की जा रही अभिव्यक्ति का मूल्यांकन कई बार किया जाएगा, क्योंकि यह वास्तव में मूल्यांकन हो सकता है कई बार। जब यह एक चर, या एक स्थिर, या एक स्तंभ संदर्भ है, तो यह एक वास्तविक समस्या होने की संभावना नहीं है; हालांकि, चीजें तेजी से बदल सकती हैं जब यह एक गैर-नियतात्मक कार्य है। विचार करें कि यह अभिव्यक्ति एक SMALLINT उत्पन्न करती है 1 और 3 के बीच; आगे बढ़ें और इसे कई बार चलाएं, और आपको हमेशा उन तीन मानों में से एक मिलेगा:
SELECT CONVERT(SMALLINT, 1+RAND()*3);
अब, इसे एक साधारण CASE में डालें अभिव्यक्ति, और इसे एक दर्जन बार चलाएं - अंततः आपको NULL . का परिणाम मिलेगा :
SELECT [result] = CASE CONVERT(SMALLINT, 1+RAND()*3) WHEN 1 THEN 'one' WHEN 2 THEN 'two' WHEN 3 THEN 'three' END;
यह कैसे होता है? खैर, पूरा CASE एक्सप्रेशन को खोजे गए एक्सप्रेशन तक विस्तृत किया जाता है, जो इस प्रकार है:
SELECT [result] = CASE WHEN CONVERT(SMALLINT, 1+RAND()*3) = 1 THEN 'one' WHEN CONVERT(SMALLINT, 1+RAND()*3) = 2 THEN 'two' WHEN CONVERT(SMALLINT, 1+RAND()*3) = 3 THEN 'three' ELSE NULL -- this is always implicitly there END;
बदले में, क्या होता है कि प्रत्येक WHEN क्लॉज RAND() का मूल्यांकन और आह्वान करता है स्वतंत्र रूप से - और प्रत्येक मामले में यह एक अलग मूल्य प्राप्त कर सकता है। मान लें कि हम व्यंजक दर्ज करते हैं, और हम पहले WHEN . की जांच करते हैं खंड, और परिणाम 3 है; हम उस खंड को छोड़ देते हैं और आगे बढ़ते हैं। यह बोधगम्य है कि RAND() . होने पर अगले दो खंड दोनों 1 लौटाएंगे फिर से मूल्यांकन किया जाता है - जिस स्थिति में किसी भी स्थिति का मूल्यांकन सही नहीं किया जाता है, इसलिए ELSE लेता है।
अन्य अभिव्यक्तियों का मूल्यांकन एक से अधिक बार किया जा सकता है
यह समस्या RAND() . तक सीमित नहीं है समारोह। इन गतिमान लक्ष्यों से आने वाली गैर-नियतात्मकता की उसी शैली की कल्पना करें:
SELECT [crypt_gen] = 1+ABS(CRYPT_GEN_RANDOM(10) % 20), [newid] = LEFT(NEWID(),2), [checksum] = ABS(CHECKSUM(NEWID())%3);
यदि कई बार मूल्यांकन किया जाए तो ये भाव स्पष्ट रूप से एक अलग मूल्य प्राप्त कर सकते हैं। और एक खोजे गए CASE . के साथ अभिव्यक्ति, ऐसे समय होंगे जब प्रत्येक पुनर्मूल्यांकन वर्तमान WHEN के लिए विशिष्ट खोज से बाहर हो जाएगा , और अंत में ELSE . दबाएं खंड। इससे खुद को बचाने के लिए, एक विकल्प यह है कि आप अपने खुद के स्पष्ट ELSE . को हमेशा हार्ड-कोड करें; केवल उस फ़ॉलबैक मान के बारे में सावधान रहें जिसे आप वापस करना चाहते हैं, क्योंकि यदि आप समान वितरण की तलाश में हैं तो इसका कुछ तिरछा प्रभाव होगा। एक अन्य विकल्प केवल अंतिम WHEN . को बदलना है ELSE . का खंड , लेकिन यह अभी भी असमान वितरण को बढ़ावा देगा। पसंदीदा विकल्प, मेरी राय में, एक बार स्थिति का मूल्यांकन करने के लिए SQL सर्वर को आज़माना और मजबूर करना है (हालाँकि यह एक ही क्वेरी के भीतर हमेशा संभव नहीं होता है)। उदाहरण के लिए, इन दो परिणामों की तुलना करें:
-- Query A: expression referenced directly in CASE; no ELSE:
SELECT x, COUNT(*) FROM
(
SELECT x = CASE ABS(CHECKSUM(NEWID())%3)
WHEN 0 THEN '0' WHEN 1 THEN '1' WHEN 2 THEN '2' END
FROM sys.all_columns
) AS y GROUP BY x;
-- Query B: additional ELSE clause:
SELECT x, COUNT(*) FROM
(
SELECT x = CASE ABS(CHECKSUM(NEWID())%3)
WHEN 0 THEN '0' WHEN 1 THEN '1' WHEN 2 THEN '2' ELSE '2' END
FROM sys.all_columns
) AS y GROUP BY x;
-- Query C: Final WHEN converted to ELSE:
SELECT x, COUNT(*) FROM
(
SELECT x = CASE ABS(CHECKSUM(NEWID())%3)
WHEN 0 THEN '0' WHEN 1 THEN '1' ELSE '2' END
FROM sys.all_columns
) AS y GROUP BY x;
-- Query D: Push evaluation of NEWID() to subquery:
SELECT x, COUNT(*) FROM
(
SELECT x = CASE x WHEN 0 THEN '0' WHEN 1 THEN '1' WHEN 2 THEN '2' END
FROM
(
SELECT x = ABS(CHECKSUM(NEWID())%3) FROM sys.all_columns
) AS x
) AS y GROUP BY x; वितरण:
| मान | <थ>प्रश्न ए <थ>प्रश्न बी <थ>प्रश्न सी <थ>क्वेरी डी||||
|---|---|---|---|---|
| शून्य | 2,572 | – | – | – |
| 0 | 2,923 | 2,900 | 2,928 | 2,949 |
| 1 | 1,946 | 1,959 | 1,927 | 2,896 |
| 2 | 1,295 | 3,877 | 3,881 | 2,891 |
विभिन्न क्वेरी तकनीकों के साथ मानों का वितरण
इस मामले में, मैं इस तथ्य पर भरोसा कर रहा हूं कि SQL सर्वर ने सबक्वायरी में अभिव्यक्ति का मूल्यांकन करने के लिए चुना है और इसे खोजे गए CASE में पेश नहीं किया है। अभिव्यक्ति, लेकिन यह केवल यह प्रदर्शित करने के लिए है कि वितरण को और भी अधिक करने के लिए मजबूर किया जा सकता है। वास्तव में, यह हमेशा अनुकूलक द्वारा चुना गया विकल्प नहीं हो सकता है, इसलिए कृपया इस छोटी सी चाल से न सीखें। :-)
CHOOSE() भी प्रभावित होता है
आप देखेंगे कि यदि आप CHECKSUM(NEWID()) . को प्रतिस्थापित करते हैं RAND() . के साथ एक्सप्रेशन अभिव्यक्ति, आपको पूरी तरह से अलग परिणाम मिलेंगे; सबसे विशेष रूप से, बाद वाला केवल एक मान लौटाएगा। ऐसा इसलिए है क्योंकि RAND() , जैसे GETDATE() और कुछ अन्य अंतर्निहित कार्यों को रनटाइम स्थिरांक के रूप में विशेष उपचार दिया जाता है, और केवल एक बार प्रति संदर्भ का मूल्यांकन किया जाता है पूरी पंक्ति के लिए। ध्यान दें कि यह अभी भी NULL लौटा सकता है पिछले कोड नमूने में पहली क्वेरी की तरह।
यह समस्या भी CASE . तक ही सीमित नहीं है अभिव्यक्ति; आप अन्य अंतर्निहित कार्यों के साथ समान व्यवहार देख सकते हैं जो समान अंतर्निहित शब्दार्थ का उपयोग करते हैं। उदाहरण के लिए, CHOOSE अधिक विस्तृत खोज के लिए केवल वाक्यात्मक चीनी है CASE अभिव्यक्ति, और इससे NULL भी प्राप्त होगा कभी-कभी:
SELECT [choose] = CHOOSE(CONVERT(SMALLINT, 1+RAND()*3),'one','two','three');
IIF() एक ऐसा फ़ंक्शन है जिसकी मुझे इसी जाल में पड़ने की उम्मीद थी, लेकिन यह फ़ंक्शन वास्तव में केवल एक खोजा गया CASE है केवल दो संभावित परिणामों के साथ अभिव्यक्ति, और कोई ELSE - इसलिए यह कठिन है, बिना नेस्टिंग और अन्य कार्यों को शुरू किए, एक ऐसे परिदृश्य की कल्पना करना जहां यह अप्रत्याशित रूप से टूट सकता है। जबकि साधारण मामले में यह CASE . के लिए अच्छा शॉर्टहैंड है , यदि आपको दो से अधिक संभावित परिणामों की आवश्यकता है तो इसके साथ कुछ भी उपयोगी करना कठिन है। :-)
COALESCE() भी प्रभावित होता है
अंत में, हमें यह जांचना चाहिए कि COALESCE समान मुद्दे हो सकते हैं। आइए मान लें कि ये भाव समान हैं:
SELECT COALESCE(@variable, 'constant'); SELECT CASE WHEN @variable IS NOT NULL THEN @variable ELSE 'constant' END);
इस मामले में, @variable दो बार मूल्यांकन किया जाएगा (जैसा कि इस कनेक्ट आइटम में वर्णित कोई फ़ंक्शन या सबक्वेरी होगा)।
जब मैं हाल ही में एक फोरम चर्चा में निम्नलिखित उदाहरण लाया तो मैं वास्तव में कुछ परेशान दिखने में सक्षम था। मान लीजिए कि मैं 1-5 से मूल्यों के वितरण के साथ एक तालिका को पॉप्युलेट करना चाहता हूं, लेकिन जब भी 3 का सामना करना पड़ता है, तो मैं इसके बजाय -1 का उपयोग करना चाहता हूं। बहुत वास्तविक दुनिया का परिदृश्य नहीं है, लेकिन निर्माण और अनुसरण करना आसान है। इस व्यंजक को लिखने का एक तरीका यह है:
SELECT COALESCE(NULLIF(CONVERT(SMALLINT,1+RAND()*5),3),-1);
(अंग्रेज़ी में, अंदर से बाहर काम करना:अभिव्यक्ति के परिणाम को रूपांतरित करें 1+RAND()*5 एक छोटे से करने के लिए; यदि उस रूपांतरण का परिणाम 3 है, तो उसे NULL . पर सेट करें; यदि उसका परिणाम NULL . है , इसे -1 पर सेट करें। आप इसे और अधिक क्रिया के साथ लिख सकते हैं CASE अभिव्यक्ति, लेकिन संक्षेप में राजा लगता है।)
यदि आप इसे कई बार चलाते हैं, तो आपको 1-5 के साथ-साथ -1 के मानों की श्रेणी देखनी चाहिए। आप 3 के कुछ उदाहरण देखेंगे, और आपने यह भी देखा होगा कि आप कभी-कभी NULL देखते हैं , हालांकि आप इनमें से किसी भी परिणाम की अपेक्षा नहीं कर सकते हैं। आइए वितरण की जांच करें:
USE tempdb; GO CREATE TABLE dbo.dist(TheNumber SMALLINT); GO INSERT dbo.dist(TheNumber) SELECT COALESCE(NULLIF(CONVERT(SMALLINT,1+RAND()*5),3),-1); GO 10000 SELECT TheNumber, occurences = COUNT(*) FROM dbo.dist GROUP BY TheNumber ORDER BY TheNumber; GO DROP TABLE dbo.dist;
परिणाम (आपके परिणाम निश्चित रूप से भिन्न होंगे, लेकिन मूल प्रवृत्ति समान होनी चाहिए):
| संख्या | घटनाएं |
|---|---|
| NULL | 1,654 |
| -1 | 2,002 |
| 1 | 1,290 |
| 2 | 1,266 |
| 3 | 1,287 |
| 4 | 1,251 |
| 5 | 1,250 |
COALESCE का उपयोग करके TheNumber का वितरण
किसी खोजे गए CASE एक्सप्रेशन को तोड़ना
क्या आप अभी तक अपना सिर खुजला रहे हैं? मान NULL . कैसे करते हैं और 3 दिखाई देते हैं, और NULL . के लिए वितरण क्यों है? और -1 काफी अधिक? ठीक है, मैं सीधे पूर्व का उत्तर दूंगा, और बाद के लिए परिकल्पनाओं को आमंत्रित करूंगा।
RAND() . के बाद से, व्यंजक मोटे तौर पर तार्किक रूप से निम्नलिखित तक विस्तृत होता है NULLIF . के अंदर दो बार मूल्यांकन किया जाता है , और उसके बाद COALESCE . की प्रत्येक शाखा के लिए दो मूल्यांकनों से गुणा करें समारोह। मेरे पास डीबगर आसान नहीं है, इसलिए यह जरूरी नहीं है कि *बिल्कुल* SQL सर्वर के अंदर क्या किया जाता है, लेकिन यह बिंदु को समझाने के लिए पर्याप्त समकक्ष होना चाहिए:
SELECT
CASE WHEN
CASE WHEN CONVERT(SMALLINT,1+RAND()*5) = 3 THEN NULL
ELSE CONVERT(SMALLINT,1+RAND()*5)
END
IS NOT NULL
THEN
CASE WHEN CONVERT(SMALLINT,1+RAND()*5) = 3 THEN NULL
ELSE CONVERT(SMALLINT,1+RAND()*5)
END
ELSE -1
END
END
तो आप देख सकते हैं कि कई बार मूल्यांकन किए जाने पर शीघ्र ही अपनी खुद की साहसिक ™ पुस्तक चुनें, और कैसे दोनों NULL और 3 संभावित परिणाम हैं जो मूल कथन की जांच करते समय संभव नहीं लगते हैं। एक दिलचस्प पक्ष नोट:यदि आप उपरोक्त वितरण स्क्रिप्ट को लेते हैं और COALESCE को प्रतिस्थापित करते हैं तो ऐसा बिल्कुल नहीं होता है ISNULL . के साथ . उस स्थिति में, NULL . की कोई संभावना नहीं है आउटपुट; वितरण मोटे तौर पर इस प्रकार है:
| संख्या | घटनाएं |
|---|---|
| -1 | 1,966 |
| 1 | 1,585 |
| 2 | 1,644 |
| 3 | 1,573 |
| 4 | 1,598 |
| 5 | 1,634 |
ISNULL का उपयोग करके TheNumber का वितरण
फिर, आपके वास्तविक परिणाम निश्चित रूप से भिन्न होंगे, लेकिन बहुत अधिक नहीं होने चाहिए। मुद्दा यह है कि हम अभी भी देख सकते हैं कि 3 अक्सर दरारों से गुजरते हैं, लेकिन ISNULL NULL . की संभावना को जादुई रूप से समाप्त कर देता है इसे पूरा करने के लिए।
मैंने COALESCE . के बीच कुछ अन्य अंतरों के बारे में बात की और ISNULL एक टिप में, जिसका शीर्षक है "SQL सर्वर में COALESCE और ISNULL के बीच निर्णय लेना।" जब मैंने यह लिखा था, तो मैं COALESCE . का उपयोग करने के पक्ष में था उस मामले को छोड़कर जहां पहला तर्क एक सबक्वेरी था (फिर से, इस बग . के कारण "फीचर गैप")। अब मुझे इतना यकीन नहीं है कि मैं इसके बारे में उतना ही दृढ़ता से महसूस करता हूं।
सिंपल CASE एक्सप्रेशन लिंक किए गए सर्वर पर नेस्ट हो सकते हैं
CASE . की कुछ सीमाओं में से एक अभिव्यक्ति यह है कि 10 घोंसले के स्तर तक सीमित है। इस उदाहरण में dba.stackexchange.com पर, पॉल व्हाइट प्रदर्शित करता है (प्लान एक्सप्लोरर का उपयोग करके) कि इस तरह की एक सरल अभिव्यक्ति:
SELECT CASE column_name WHEN '1' THEN 'a' WHEN '2' THEN 'b' WHEN '3' THEN 'c' ... END FROM ...
पार्सर द्वारा खोजे गए फ़ॉर्म में विस्तृत हो जाता है:
SELECT CASE WHEN column_name = '1' THEN 'a' WHEN column_name = '2' THEN 'b' WHEN column_name = '3' THEN 'c' ... END FROM ...
लेकिन वास्तव में एक लिंक किए गए सर्वर कनेक्शन पर निम्नलिखित के रूप में प्रेषित किया जा सकता है, और अधिक वर्बोज़ क्वेरी:
SELECT
CASE WHEN column_name = '1' THEN 'a' ELSE
CASE WHEN column_name = '2' THEN 'b' ELSE
CASE WHEN column_name = '3' THEN 'c' ELSE
...
ELSE NULL
END
END
END
FROM ...
इस स्थिति में, भले ही मूल क्वेरी में केवल एक ही CASE था 10+ संभावित परिणामों के साथ अभिव्यक्ति, जब लिंक किए गए सर्वर पर भेजा गया, तो इसमें 10+ नेस्टेड . था CASE भाव। जैसे, जैसा कि आप उम्मीद कर सकते हैं, इसने एक त्रुटि लौटा दी:
विवरण (विवरण) तैयार नहीं किए जा सके।
संदेश 125, स्तर 15, राज्य 4
मामले के भाव केवल स्तर 10 में नेस्ट किए जा सकते हैं।
कुछ मामलों में, आप इसे पॉल के सुझाव के अनुसार फिर से लिख सकते हैं, इस तरह की अभिव्यक्ति के साथ (column_name मानते हुए) एक वर्चर कॉलम है):
SELECT CASE CONVERT(VARCHAR(MAX), SUBSTRING(column_name, 1, 255)) WHEN 'a' THEN '1' WHEN 'b' THEN '2' WHEN 'c' THEN '3' ... END FROM ...
कुछ मामलों में, केवल SUBSTRING उस स्थान को बदलने की आवश्यकता हो सकती है जहां अभिव्यक्ति का मूल्यांकन किया जाता है; अन्य में, केवल CONVERT . मैंने संपूर्ण परीक्षण नहीं किया, लेकिन यह लिंक किए गए सर्वर प्रदाता, कोलेशन कम्पैटिबल और यूज़ रिमोट कॉलेशन जैसे विकल्पों और पाइप के दोनों छोर पर SQL सर्वर के संस्करण के साथ करना पड़ सकता है।
लंबी कहानी छोटी, यह याद रखना महत्वपूर्ण है कि आपका CASE अभिव्यक्ति को बिना किसी चेतावनी के आपके लिए फिर से लिखा जा सकता है, और यह कि आपके द्वारा उपयोग किए जाने वाले किसी भी समाधान को बाद में अनुकूलक द्वारा खारिज किया जा सकता है, भले ही वह अभी आपके लिए काम करता हो।
केस एक्सप्रेशन अंतिम विचार और अतिरिक्त संसाधन
मुझे आशा है कि मैंने CASE के कुछ कम-ज्ञात पहलुओं पर विचार करने के लिए कुछ भोजन दिया है अभिव्यक्ति, और परिस्थितियों में कुछ अंतर्दृष्टि जहां CASE - और कुछ फ़ंक्शन जो समान अंतर्निहित तर्क का उपयोग करते हैं - अप्रत्याशित परिणाम लौटाते हैं। कुछ अन्य दिलचस्प परिदृश्य जहां इस प्रकार की समस्या सामने आई है:
- स्टैक ओवरफ्लो :यह केस एक्सप्रेशन ELSE क्लॉज तक कैसे पहुंचता है?
- स्टैक ओवरफ़्लो:CRYPT_GEN_RANDOM () अजीब प्रभाव
- स्टैक ओवरफ़्लो:चुनें () इरादे के अनुसार काम नहीं कर रहा है
- स्टैक ओवरफ़्लो:CHECKSUM(NewId()) प्रति पंक्ति कई बार निष्पादित करता है
- कनेक्ट #350485 :NEWID() और टेबल एक्सप्रेशन के साथ बग