वचन तोड़े गए?
त्रुटियाँ अनदेखी कर दी गई हैं, परिणाम खो देना...
जावास्क्रिप्ट प्रमिस टूटे हुए हैं?
पहले के दिनों में
प्रमिस के बारे में सबसे आम मिथक यह है कि इनके त्रुटि के कमजोर पक्ष का आरोप है।
कई साल पहले प्रमिस वास्तव में त्रुटियों के साथ खराब थे। इन्हें ठीक करने में बहुत मेहनत की गई।
और फिर, इसे ठीक कर दिया गया, यहां तक कि **व्यापक रूप से तैनात कर दिया गया।
लोग खुश हुए
और दुर्भाग्य से, कुछ लोग ध्यान नहीं दे पाए।
अब के दौर
मिथक अभी भी बरकरार है, मैंने इसे हर जगह देखा है: मीडियम पर लोकप्रिय लेख, dzone पर, और कई अन्य स्रोतों पर।
मैं स्वीकार करता हूं, यहां तक कि “अधिकारियों” द्वारा दिए गए संसाधनों और दस्तावेज़न भी अधिकांश कमजोर उदाहरणों और बुरी आदतों के साथ हैं। इनका उपयोग अक्सर प्रमिस के खिलाफ तर्क को साबित करने के लिए किया जाता है। कुछ लोग तो ऐसे “उपाय” भी सुझाते हैं जो चीजें बहुत खराब कर देते हैं। (नोट: लिंक हटा दिया गया है)
नियम: समस्याओं से दूर रहने के लिए
- प्रमिस को कुछ ऐसा चाहिए जिस पर टिका रहे
- हमेशा अपने फ़ंक्शन से
returnकरें।
- हमेशा अपने फ़ंक्शन से
- वास्तविक
Errorइंस्टेंस का उपयोग करें- हमेशा
Errorइंस्टेंस का उपयोग करें।
- हमेशा
- उस जगह त्रुटियाँ संभालें जहाँ यह बनता है
- हमेशा
.catch()का उपयोग करें, कम से कम एक बार।
- हमेशा
- नामित फ़ंक्शन के साथ स्पष्टता जोड़ें 🦄✨
- प्राथमिकता दें नामित फ़ंक्शन को।
#1 प्रमिस को कुछ ऐसा चाहिए जिस पर टिका रहे
यह आवश्यक है कि आप हमेशा return करें अपने फ़ंक्शन से।
.then(callback) और .catch(callback) में प्रमिस कॉलबैक फ़ंक्शन एक निश्चित पैटर्न के अनुसार काम करते हैं।
प्रत्येक वापसी मूल्य अगले .then() के कॉलबैक में पास हो जाता है।
function addTen(number) { return number + 10;}
Promise.resolve(10) // 10 .then(addTen) // 20 .then(addTen) // 30 .then(addTen) // 40 .then(console.log) // logs "40"“हमेशा वापसी करने” का लाभ: यूनिट टेस्ट करना बहुत आसान हो जाता है।
प्रश्न: कितने अलग-अलग प्रमिस स्थितियाँ (सुलझाए गए & अस्वीकृत) बनाई गईं?
प्रश्न: पिछले उदाहरण में कितने प्रमिस बनाए गए थे?
#2 वास्तविक Error उदाहरणों का उपयोग करें
जावास्क्रिप्ट में त्रुटियों के आसपास एक रोचक व्यवहार है (जो असिंक्रोनस और सिंक्रोनस दोनों कोड में लागू होता है।)
repl.it में उदाहरण देखें: throwing errors in javascript
कॉल स्टैक और लाइन नंबर के बारे में उपयोगी विवरण प्राप्त करने के लिए, आपको Error इंस्टेंस का उपयोग करना होगा। स्ट्रिंग फेंकने काम नहीं करते हैं जैसे कि पायथन या रूबी में।
जावास्क्रिप्ट सीधा तौर पर throw "string" को संभालता है, आप अपने catch हैंडलर में स्ट्रिंग देखेंगे। हालांकि, डेटा ही आपको दिखाई देगा। कोई पूर्व स्टैक फ्रेम शामिल नहीं होगा।
सही new Error उदाहरण:
throw new Error('message') // ✅Promise.reject(new Error('message')) // ✅throw Error('message') // ✅Promise.reject(Error('message')) // ✅निम्नलिखित आम अनुचित पैटर्न हैं:
throw 'error message' // ❌Promise.reject(-42) // ❌#3 त्रुटियों का उपयोग जहां बनता है वहां करें
प्रमिस त्रुटियों के उपयोग के लिए एक सुंदर तरीका प्रदान करते हैं, .catch() का उपयोग करके। यह आधार रूप से .then() का एक विशेष प्रकार है - जहां पूर्ववर्ती .then() की कोई भी त्रुटियां नियंत्रित की जाती हैं। एक उदाहरण देखते हैं…
Promise.resolve(42) .then(() => 'hello') .catch(() => console.log('will not get hit')) .then(() => throw new Error('totes fail')) .catch(() => console.log('WILL get hit'))हालांकि .catch() एक डोम घटना हैंडलर (जैसे click, keypress) की तरह लग सकता है। इसकी स्थिति महत्वपूर्ण है, क्योंकि यह केवल ‘पकड़’ सकता है जो ऊपर फेंका गया हो।
त्रुटियों का अतिलेखन अपेक्षाकृत सरल है अपने .catch() कॉलबैक में एक गैर-त्रुटि मान लौटाएं, प्रमिस श्रृंखला अगले .then() कॉलबैक के क्रम में चलना शुरू कर देती है। (सक्रिय रूप से।)
निम्न उदाहरण के क्रम का अनुसरण करने की कोशिश करें:
Promise.resolve(42) .then(() => 'hello') .then(() => throw new Error('totes fail')) .catch(() => { return 99 }) .then(num => num + 1) .then(console.log) // अपेक्षित आउटपुट: 100क्रम को समझना महत्वपूर्ण है।
हालांकि यह एक हास्यास्पद उदाहरण है, लेकिन इसका उद्देश्य त्रुटियों और डेटा के प्रवाह को कैसे दर्शाना है।
यहां अनुक्रम का एक आउटलाइन है:
- 42 प्रारंभिक मान है।
helloअगली विधि द्वारा हमेशा लौटाया जाता है।- हम पिछले मान को अनदेखा करते हैं, और
'totes fail'संदेश के साथ एक त्रुटि फेंक देते हैं। .catch()त्रुटि को अवरुद्ध करता है, बजाय99को लौटाता है जो कि किसी भी अगले.then()द्वारा संसाधित होगा।numको बढ़ाएं,100लौटाते हैं- विधि
console.log100प्राप्त करती है और इसे प्रिंट करती है! :tada:
प्रश्न: जब 2 .catch() क्रम में हों तो क्या होता है? क्या दूसरा कभी चल सकता है? क्या आप एक उपयोग के मामले की कल्पना कर सकते हैं?
प्रश्न: .catch() कैसे त्रुटियों को अनदेखा कर सकता है? आप कैसे त्रुटियों को Promise.all के अग्रिम निकास से रोकेंगे?
#4 नामित फ़ंक्शनों के साथ स्पष्टता जोड़ें 🦄✨
निम्नलिखित 2 उदाहरणों की पठनीयता की तुलना करें:
अनामित: ❌
Promise.resolve(10) // 10 .then(x => x * 2) // 20 .then(x => x / 4) // 5 .then(x => x * x) // 25 .then(x => x.toFixed(2)) // "25.00" .then(x => console.log(x)) // expected output: "25.00"नामित: ✅
Promise.resolve(10) // 10 .then(double) // 20 .then(quarter) // 5 .then(square) // 25 .then(format) // "25.00" .then(log) // expected output: "25.00"
const double = x => x * 2const quarter = x => x / 4const square = x => x * xconst format = x => x.toFixed(2)const log = x => console.log(x)बोनस: ✅
सरणी विधि संगत!!!
आप अपने नामित फ़ंक्शन को Array.prototype. से हमारे दोस्तों के साथ पुनः उपयोग कर सकते हैं। शामिल .map(), .filter(), .every(), .some(), .find()!
कलेक्शन पाइपलाइन्स #FTW:
// यह वही चीज़ है :mindblown:
[10, 20] // [ 10, 20 ] .map(double) // [ 20, 40 ] .map(quarter) // [ 5, 10 ] .map(square) // [ 25, 100 ] .map(format) // [ "25.00", "100.00" ] .map(log) // अपेक्षित 2 पंक्तियाँ आउटपुट: "25.00", "100.00"अगर आप ऐसा रैखिक-शैली कोडिंग नहीं करना चाहते हैं… तो आपके पास सरल फ़ंक्शन हैं!
आप जैसा चाहें उनका उपयोग कर सकते हैं:
// नेस्टिंग पैटर्न// ❌ कृपया ऐसा न करें, हालाँकि
const result = format(square(quarter(double(10))))
log(result)// अपेक्षित आउटपुट: "25.00"क्यों फ़ंक्शन नेस्टिंग एक एंटी-पैटर्न है?
- कम लोगों के लिए समझने में आसान नहीं होता है
- git डिफ़्स आसानी से यह नहीं दिखाते कि कौन क्या बदला
- नेस्टेड फ़ंक्शन्स के मध्य डीबग करना या लॉग करना कठिन होता है