जावास्क्रिप्ट भाषा में सबसे बड़े buzzwords में से एक क्लोजर है। यह FAANG कंपनियों में नौकरी के लिए इंटरव्यू के कई सवालों का विषय है। इस लेख में, हम क्लोजर और स्कोप के बारे में बात करेंगे, इसकी अवधारणाओं को सरल उदाहरणों के साथ स्पष्ट करेंगे, और फिर एक बड़े तकनीकी दिग्गज के साथ एक साक्षात्कार से एक नमूना प्रश्न के साथ समाप्त करेंगे।
दायरा
जब कोई आपको बताता है कि कुछ परियोजना के दायरे में है या नहीं, तो इसका क्या मतलब है?
जब मैं इसका उत्तर सोचता हूं तो मैं एक पेरिस्कोप या दूरबीन के बारे में सोचना चाहता हूं। ये उपकरण हमें उसके पास मौजूद लेंस की सीमा के भीतर सभी प्रकार की चीजें दिखाते हैं:यह दायरे में है . अगर यह दायरे के बाहर है , आप लेंस के व्यास से आगे नहीं देख सकते हैं। और व्यास के बाहर किसी चीज पर प्रकाश डालना संभव नहीं है। आपको इसके बारे में सोचना चाहिए क्योंकि हम जावास्क्रिप्ट में तीन बहुत महत्वपूर्ण और विशिष्ट प्रकार के दायरे के बारे में बात करते हैं:स्थानीय, वैश्विक और शाब्दिक।
स्थानीय दायरा
आज हम जिन तीन क्षेत्रों के बारे में बात करेंगे, उनमें से स्थानीय दायरा सबसे छोटा है। जब हम कोई फ़ंक्शन घोषित करते हैं, तो कोष्ठक ({}) के अंदर कुछ भी फ़ंक्शन के लिए स्थानीय माना जाता है। जब जावास्क्रिप्ट इंजन फ़ंक्शन को पढ़ता है तो यह चर घोषित करेगा; जब यह समाप्त होता है तो यह चर को नष्ट कर देगा।
function greeting() { var websiteName = 'Career Karma'; return `Hello ${websiteName}`; } console.log(greeting()); // Hello Career Karma console.log(websiteName); // ReferenceError: websiteName is not defined
जैसा कि आप देख सकते हैं, जब हम बुलाए गए ग्रीटिंग फ़ंक्शन के परिणाम "console.log ()" करते हैं, तो हम फ़ंक्शन निष्पादित होने के बाद वेबसाइटनाम तक पहुंचने में सक्षम होते हैं। यह हमें 'हैलो करियर कर्मा' स्ट्रिंग देता है जिसकी हमें तलाश थी। console.log()
फ़ंक्शन के अंदर घोषित किया गया चर एक त्रुटि फेंकता है क्योंकि यह परिभाषित नहीं है।
जैसा कि पहले ही उल्लेख किया गया है, वेबसाइटनाम अपरिभाषित क्यों है, क्योंकि वेरिएबल फ़ंक्शन के अंदर बनाए जाते हैं जब उन्हें बुलाया जाता है और फिर टर्मिनल स्टेटमेंट चलने पर नष्ट हो जाता है। फ़ंक्शन के बाहर किसी भी चीज़ की फ़ंक्शन के अंदर की सामग्री तक पहुंच नहीं होती है, जब तक कि उसके पास कोई विशेष सेटअप न हो।
वैश्विक दायरा
यह अगला दायरा काफी हद तक वाक्यांश का शाब्दिक अनुवाद है। एक वैश्विक दायरा एक समारोह के बाहर घोषित वस्तुओं को लेता है और उन्हें एक ऐसे स्थान पर सुरक्षित रखता है जहां सभी स्क्रिप्ट और विधियां और कार्य उनके तर्क के लिए उनका उपयोग और उपयोग कर सकते हैं।
let counter = 0; // global -- declared outside function const add = () => { // function declaration let counter = 0; // local -- declared inside function counter += 1; // counter increased by 1 -- which counter variable increased? return counter; } add(); // invoke add(); // three add(); // times console.log(counter) // is this 3 or 0? Why?
यदि हम console.log()
कोड के अंत में काउंटर? आप क्या होने की उम्मीद करते हैं?
81% प्रतिभागियों ने कहा कि बूटकैंप में भाग लेने के बाद उन्हें अपनी तकनीकी नौकरी की संभावनाओं के बारे में अधिक आत्मविश्वास महसूस हुआ। आज ही एक बूटकैंप से मिलान करें।
बूटकैंप शुरू करने से लेकर अपनी पहली नौकरी खोजने तक, औसत बूटकैंप ग्रेड ने करियर संक्रमण में छह महीने से भी कम समय बिताया।
आइए कोड के माध्यम से चलते हैं:
- वैश्विक परिवेश में काउंटर वैरिएबल घोषित और आरंभ किया गया।
- वैश्विक वातावरण में घोषित फ़ंक्शन जोड़ें।
- जोड़ना शुरू हो गया है।
- काउंटर वैरिएबल घोषित और स्थानीय वातावरण में शुरू किया गया।
- स्थानीय काउंटर में 1 की वृद्धि होती है, स्थानीय और वैश्विक क्यों नहीं?
- काउंटर लौटा दिया गया है। समारोह समाप्त हो जाता है।
- जोड़ें फिर से लागू हो गई हैं
- चरण 4 से 6 तक फिर से चलें।
- चरण 3 से 6 फिर से दोहराएं।
console.log(counter)
; क्या लौटाया गया है?
क्योंकि फ़ंक्शन समाप्त हो जाता है जब काउंटर हर बार 1 पर होता है, हमारे स्थानीय काउंटर वैरिएबल को हर बार फ़ंक्शन के चलने पर 0 पर फिर से शुरू किया जाता है। चाहे कुछ भी हो जाए, स्थानीय स्तर पर काउंटर हमेशा 1 पर रुकेगा।
यदि कोई फ़ंक्शन अपने दायरे में एक चर पाता है, तो यह चर के वैश्विक दायरे को नहीं देखता है - इसलिए वैश्विक चर कभी नहीं बदलता है। तो, हमारा console.log()
आउटपुट 0 होगा क्योंकि उस कथन के वातावरण के भीतर हमारा निकटतम परिभाषित चर वैश्विक वातावरण में है।
लेक्सिकल स्कोप
लेक्सिकल स्कोप जावास्क्रिप्ट में सबसे मौलिक अवधारणाओं में से एक है। यह विचार है कि किसी फ़ंक्शन या चर का निर्माण कोड के कुछ हिस्सों तक पहुंच योग्य होगा और फिर कोड के अन्य हिस्सों तक पहुंच योग्य नहीं होगा। यह सब इस बात पर निर्भर करता है कि प्रत्येक चर और कार्य की घोषणा कहाँ है।
आइए कोड के इस ब्लॉक पर एक नज़र डालें:
const init = () => { // <== This is our outer function const var1 = 'Career'; // outer scope const second = () => { // <== This is our inner function const var2 = 'Karma'; // inner scope console.log(var1); // Career console.log(var2); // Karma return var1 + " " + var2; }; // console.log(var2); // undefined return second(); }; init();
यहां हमारे पास नेस्टेड फ़ंक्शंस का एक सेट है। init()
फ़ंक्शन var1 नामक एक चर घोषित करता है, दूसरे नामक फ़ंक्शन की घोषणा करता है और second()
. को आमंत्रित करता है .
जब कंपाइलर पहली बार इस कोड से गुजरता है, तो यह हमारे पास जो कुछ है उस पर एक उच्च स्तरीय नज़र डालता है:
init()
समारोह- आह्वान
init()
इस बिंदु पर, हम init () फ़ंक्शन के अंदर और कुछ नहीं देख सकते हैं - हम जानते हैं कि फ़ंक्शन मौजूद है। जब हमारे init () func को लागू किया जाता है, तो कंपाइलर फ़ंक्शन के अंदर क्या है, इस पर एक और उच्च स्तरीय नज़र डालता है:
var1
second()
समारोह- आह्वान
second()
init()
फ़ंक्शन के बारे में कुछ भी नहीं जानता है कि second()
. के अंदर क्या हो रहा है खंड मैथा। यह केवल वही देख सकता है जो इसके व्याख्यात्मक वातावरण . में है - इसके आसपास की स्थिति।
प्रत्येक नेस्टेड फ़ंक्शन एक छोटे कंटेनर में होता है, जैसे उन रूसी matryoshka नेस्टिंग गुड़िया का एक सेट (उदाहरण के लिए पृष्ठ के शीर्ष देखें यदि आप सुनिश्चित नहीं हैं कि वे क्या हैं)। गुड़िया केवल इस बारे में जानती हैं कि उनके कंटेनर के अंदर क्या चल रहा है और माता-पिता में पहले से क्या हुआ या घोषित / पढ़ा गया है। उदाहरण के लिए सबसे बड़ी गुड़िया केवल यह जानती है कि उसके कंटेनर में अगली गुड़िया मौजूद है। यह सेट में किसी भी अन्य गुड़िया के बारे में नहीं जानता है, बस इसके शाब्दिक वातावरण (इसकी स्थिति) में क्या है और पहले से क्या हुआ है (बाहरी दायरा)।
संक्षेप में, हम दो बातें जानते हैं:
- बाहरी दायरा भीतरी दायरा नहीं देख सकता।
- आंतरिक दायरे की बाहरी दायरे तक पहुंच है।
क्योंकि बाहरी सामना नहीं देख सकता कि आंतरिक दायरे में क्या हो रहा है, हम सुरक्षित रूप से कह सकते हैं कि यह एकतरफा संबंध है। आंतरिक बाहरी दायरे से चर देख और उपयोग कर सकता है, लेकिन बाहरी आंतरिक नहीं देख सकता है। इसे लेक्सिकल स्कोप . कहा जाता है .
लेक्सिकल स्कोपिंग की सुंदरता यह है कि एक चर का मान कोड में उसके स्थान से निर्धारित होता है। फ़ंक्शन पहले अपने स्थानीय वातावरण के अंदर एक चर के अर्थ की तलाश करते हैं - यदि वह इसे नहीं ढूंढ पाता है, तो यह उस फ़ंक्शन पर चला जाता है जो उस फ़ंक्शन को परिभाषित करता है। यदि यह वहां नहीं मिलता है, तो यह श्रृंखला को अगले परिभाषित फ़ंक्शन तक ले जाता है।
यह जावास्क्रिप्ट में एक बहुत ही महत्वपूर्ण अवधारणा बन जाती है जो बार-बार सामने आएगी जब आप जावास्क्रिप्ट फ्रेमवर्क के बारे में अधिक जानेंगे और वे कैसे काम करेंगे। आप बाहर से नीचे जा सकते हैं, लेकिन आप दूसरी दिशा में कभी भी "ऊपर" नहीं जा सकते। यह बहुत महत्वपूर्ण है क्योंकि हम मुख्य विषय पर आते हैं:बंद करें ।
बंद
क्लोजर की परिभाषा लेक्सिकल स्कोप से काफी मिलती-जुलती है। दोनों के बीच मुख्य अंतर यह है कि क्लोजर एक उच्च क्रम का कार्य है और लेक्सिकल स्कोपिंग नहीं है। एक उच्च क्रम फ़ंक्शन में एक मूल विशेषता होती है:यह या तो एक फ़ंक्शन देता है या किसी फ़ंक्शन को पैरामीटर के रूप में उपयोग करता है।
क्लोजर एक ऐसा फंक्शन है जो अपने लेक्सिकल स्कोप तक पहुंच सकता है, तब भी जब उस फंक्शन को बाद में लागू किया जा रहा हो।
क्लोजर और लेक्सिकल स्कोप दोनों के अपने वेरिएबल हैं, पैरेंट फंक्शन के वेरिएबल्स और पैरामीटर्स तक पहुंच सकते हैं, और ग्लोबल वेरिएबल्स का उपयोग कर सकते हैं। आइए निम्नलिखित कोड के माध्यम से चलते हैं:
function greeting() { //outer scope (parent function) const userName = "CrrKrma1952"; // parent variable function welcomeGreeting() { // inner function console.log("Hello, " + userName); // accesses parent var return "Hello, " + userName; // terminal statement } return welcomeGreeting; // returns a function (which makes it HOF) } // end of greeting() const greetUser = greeting(); // greetUser(); // Hello, CrrKrma1952
greeting()
फ़ंक्शन मौजूद है, लेकिन हम अभी तक सामग्री को नहीं जानते हैं।greetUser
मौजूद है, लेकिन अभी तक सामग्री नहीं जानताgreetUser()
- यह पिछली पंक्ति को आमंत्रित करता है, जो बदले में, ग्रीटिंग () फ़ंक्शन को आमंत्रित करता है।- उपयोगकर्ता नाम घोषित
welcomeGreeting()
मौजूद है, लेकिन अभी तक सामग्री नहीं जानता- वापसी विवरण
welcomeGreeting()
. के नीचे ब्लॉक वही फ़ंक्शन देता है console.log(‘Hello, ‘ + userName)
; यहां हमारा कंसोल.लॉग उपयोगकर्ता नाम का मान प्राप्त करने के लिए पैरेंट स्कोप तक पहुंच सकता है- टर्मिनल स्टेटमेंट जो फ़ंक्शन को समाप्त करता है और कोड ब्लॉक के अंदर चर के अर्थ को नष्ट कर देता है।
इस कोड में, हम एक साथ नेस्टिंग फंक्शन द्वारा सूचनाओं को प्रसारित कर रहे हैं ताकि बाद में पैरेंट स्कोप तक पहुँचा जा सके।
निष्कर्ष
इस लेख में, हमने एक बहुत ही भारी जावास्क्रिप्ट विषय के बारे में बात की:स्कोप और क्लोजर। मैं इस विषय पर कई अलग-अलग लेख पढ़ने और पढ़ने की सलाह दूंगा। जिस तरह से इसे पढ़ाया जाता है वह विभिन्न दृष्टिकोणों से आ सकता है - जिसका अर्थ है, निश्चित रूप से, इसे सीखने के बहुत सारे तरीके हैं। मुझे आशा है कि यह प्राइमर आपके लिए मददगार था! क्लोजर पर अपना अध्ययन जारी रखने के लिए शुभकामनाएँ!