State as a Snapshot
State वेरिएबल नियमित जावास्क्रिप्ट वेरिएबलों की तरह दिख सकते हैं जिन्हें आप पढ़ और लिख सकते हैं। हालांकि, state एक स्नैपशॉट की तरह व्यवहार करता है। इसे सेट करने से पहले आपके पास पहले से मौजूद state वेरिएबल में कोई परिवर्तन नहीं होता है, बल्कि इसे फिर से रेंडर करने को ट्रिगर करता है।
You will learn
- State सेट करने से री-रेंडर को ट्रिगर कैसे किया जाता है
- State कब और कैसे अपडेट होता है
- State को सेट करने के तत्काल बाद क्यों तुरंत अपडेट नहीं होता
- इवेंट हैंडलर्स कैसे State के “स्नैपशॉट” तक पहुंच प्राप्त करते हैं
Setting state triggers renders
आप अपना यूज़र इंटरफ़ेस को सीधे उपयोगकर्ता की क्लिक जैसी घटना के प्रतिक्रिया के प्रति सोच सकते हैं। React में, यह मेंटल मॉडल से थोड़ा अलग काम करता है। पिछले पेज पर, आपने देखा कि React state सेट करने से री-रेंडर का अनुरोध होता है। इसका मतलब है कि एक इंटरफ़ेस को घटना के प्रतिक्रिया के लिए, आपको state को अपडेट करने की आवश्यकता होती है।
इस उदाहरण में, जब आप “send” दबाते हैं, setIsSent(true)
React को यह सूचित करता है कि UI को री-रेंडर किया जाए।
import { useState } from 'react'; export default function Form() { const [isSent, setIsSent] = useState(false); const [message, setMessage] = useState('नमस्ते!'); if (isSent) { return <h1>आपका संदेश पहुंचने की प्रक्रिया जारी है!</h1> } return ( <form onSubmit={(e) => { e.preventDefault(); setIsSent(true); sendMessage(message); }}> <textarea placeholder="Message" value={message} onChange={e => setMessage(e.target.value)} /> <button type="submit">Send</button> </form> ); } function sendMessage(message) { // ... }
बटन पर क्लिक करने पर निम्नलिखित होता है:
onSubmit
घटना हैंडलर क्रियान्वित होता है।setIsSent(true)
द्वाराisSent
कोtrue
में सेट किया जाता है और एक नया रेंडर करने की क्रिया शुरू हो जाती है।- React नए
isSent
वैल्यू के आधार पर कंपोनेंट को फिर से रेंडर करता है।
चलिए state और रेंडरिंग के बीच का संबंध और विश्लेषण करें।
Rendering takes a snapshot in time
रेंडरिंग का मतलब है कि React आपके कंपोनेंट को कॉल करता है, जो एक फ़ंक्शन होता है। आप उस फ़ंक्शन से जो JSX रिटर्न करते हैं, वह उस समय की UI का एक स्नैपशॉट होता है। इसमें props, इवेंट हैंडलर्स और स्थानिक वेरिएबल इत्यादि सभी वैल्यूें, रेंडर करने के समय के state के अनुसार निर्धारित की गई थीं।
एक फ़ोटोग्राफ या मूवी फ़्रेम की तरह, आपके द्वारा रिटर्न किया गया यूआई “स्नैपशॉट” इंटरैक्टिव होता है। इसमें इवेंट हैंडलर्स जैसी लॉजिक शामिल होती है जो इनपुट के प्रतिक्रिया के लिए स्पष्ट करती है। React स्क्रीन को इस स्नैपशॉट के साथ अपडेट करता है और इवेंट हैंडलर्स को कनेक्ट करता है। इस परिणामस्वरूप, बटन को दबाने से आपके JSX में से क्लिक हैंडलर ट्रिगर होगा।
जब React किसी कंपोनेंट को री-रेंडर करता है:
- React आपके फ़ंक्शन को फिर से कॉल है।
- आपका फ़ंक्शन एक नया JSX स्नैपशॉट रिटर्न करता है।
- फिर, React स्क्रीन को उस स्नैपशॉट के साथ अपडेट करता है जिसे आपने रिटर्न किया है।
Illustrated by Rachel Lee Nabors
एक कॉंपोनेंट की मेमोरी के रूप में, state एक साधारित चर की तरह नहीं है जो आपके फ़ंक्शन के लौटने के बाद गायब हो जाता है। State वास्तव में React के भीतर “रहती है” - जैसे कि एक शेल्फ पर! - आपके फ़ंक्शन के बाहर। जब React आपके कंपोनेंट को कॉल करता है, तो वह आपको उस विशेष रेंडर के लिए state का एक स्नैपशॉट देता है। आपका कंपोनेंट एक नई सेट के props और इवेंट हैंडलर के साथ UI का एक स्नैपशॉट वापस करता है, जो उस रेंडर की state मानों का उपयोग करके कॅल्क्युलेट होता है!
Illustrated by Rachel Lee Nabors
यहां एक छोटा प्रयोग है जो आपको दिखाएगा कि यह कैसे काम करता है। इस उदाहरण में, आपकी उम्मीद हो सकती है कि “+3” बटन पर क्लिक करने से काउंटर को तीन बार बढ़ाने की प्रत्येक बार setNumber(number + 1)
को कॉल के कारण होना चाहिए।
जब आप “+3” बटन दबाते हैं, तो देखिए क्या होता है:
import { useState } from 'react'; export default function Counter() { const [number, setNumber] = useState(0); return ( <> <h1>{number}</h1> <button onClick={() => { setNumber(number + 1); setNumber(number + 1); setNumber(number + 1); }}>+3</button> </> ) }
ध्यान दें कि प्रति क्लिक number केवल एक बार बढ़ता है!
State सेट करने से केवल अगले रेंडर के लिए इसे बदल दिया जाता है। पहले रेंडर के दौरान, number
की वैल्यू 0
थी। इसलिए, उस रेंडर के onClick
हैंडलर में, setNumber(number + 1)
को बुलाने के बाद भी number
का वैल्यू 0
ही है।
<button onClick={() => {
setNumber(number + 1);
setNumber(number + 1);
setNumber(number + 1);
}}>+3</button>
यहां इस बटन के क्लिक हैंडलर का कहना है कि React को क्या करना चाहिए:
setNumber(number + 1)
:number
की वैल्यू0
है, इसलिएsetNumber(0 + 1)
को करें।- React अगले रेंडर पर
number
को 1 में बदलने की तैयारी करता है।
- React अगले रेंडर पर
setNumber(number + 1)
:number
की वैल्यू0
है, इसलिएsetNumber(0 + 1)
को करें।- React अगले रेंडर पर
number
को 1 में बदलने की तैयारी करता है।
- React अगले रेंडर पर
setNumber(number + 1)
:number
की वैल्यू0
है, इसलिएsetNumber(0 + 1)
को करें।- React अगले रेंडर पर
number
को 1 में बदलने की तैयारी करता है।
- React अगले रेंडर पर
हालांकि आपने setNumber(number + 1)
को तीन बार कॉल करता है, लेकिन इस रेंडर के इवेंट हैंडलर में number
हमेशा 0
होता है, इसलिए आपने तीन बार state को 1
में सेट किया है। इसलिए, जब आपका इवेंट हैंडलर समाप्त होता है, React number
के साथ अद्यतित होकर कंपोनेंट को 3
के बजाय 1
के साथ री-रेंडर करता है।
आप इसे इस प्रकार से भी मानसिक रूप से दृश्यीकरण कर सकते हैं, अपने कोड में state वेरिएबल को उनके वैल्यूेंस के साथ प्रतिस्थापित करके। क्योंकि number
state वेरिएबल इस रेंडर के लिए 0
है, इसलिए इसके इवेंट हैंडलर का दिखने का यह तरीका है:
<button onClick={() => {
setNumber(0 + 1);
setNumber(0 + 1);
setNumber(0 + 1);
}}>+3</button>
अगले रेंडर के लिए, number
1 है, इसलिए उस रेंडर के क्लिक हैंडलर का दिखावा इस तरह होता है:
<button onClick={() => {
setNumber(1 + 1);
setNumber(1 + 1);
setNumber(1 + 1);
}}>+3</button>
इसलिए बटन पर फिर से क्लिक करने से काउंटर को 2
पर सेट किया जाएगा, फिर अगले क्लिक पर 3
पर और ऐसी ही प्रक्रिया जारी रहेगी।
State over time
अच्छा, यह बहुत मजेदार था। कोशिश करें कि इस बटन पर क्लिक करने पर क्या चेतावनी दी जाएगी, इसे अनुमान लगाएँ:
import { useState } from 'react'; export default function Counter() { const [number, setNumber] = useState(0); return ( <> <h1>{number}</h1> <button onClick={() => { setNumber(number + 5); alert(number); }}>+5</button> </> ) }
यदि आप पहले बताए गए प्रतिस्थापन विधि का उपयोग करते हैं, तो आप अनुमान लगा सकते हैं कि अलर्ट “0” दिखाता है:
setNumber(0 + 5);
alert(0);
लेकिन यदि आप चेतावनी पर टाइमर लगाएं, जिससे वह केवल कंपोनेंट री-रेंडर के बाद ही सक्रिय होती है, तो क्या यह “0” या “5” दिखाएगी? कृपया अनुमान लगाएं!
import { useState } from 'react'; export default function Counter() { const [number, setNumber] = useState(0); return ( <> <h1>{number}</h1> <button onClick={() => { setNumber(number + 5); setTimeout(() => { alert(number); }, 3000); }}>+5</button> </> ) }
हैरानी हुई? यदि आप प्रतिस्थापन विधि का उपयोग करते हैं, तो आप अलर्ट को पास किए गए state की “स्नैपशॉट” देख सकते हैं।
setNumber(0 + 5);
setTimeout(() => {
alert(0);
}, 3000);
React में संग्रहित state अलर्ट चलाने के समय बदल सकती है, लेकिन यह state का स्नैपशॉट उस समय के लिए यूजर के साथ इंटरैक्ट करने पर निर्धारित किया गया था!
रेंडर के भीतर एक state वेरिएबल का वैल्यू कभी नहीं बदलता, यद्यपि इसके इवेंट हैंडलर कोड असिंक्रोनस हो। उस रेंडर के onClick
में, number
की वैल्यू setNumber(number + 5)
को कॉल करने के बाद भी 0
ही रहती है। इसकी वैल्यू React द्वारा आपके कंपोनेंट को कॉल करके UI का “स्नैपशॉट” लेने पर “निश्चित” की गई थी।
यहां एक उदाहरण है जो दिखाता है कि यह कैसे आपके इवेंट हैंडलर को टाइमिंग गलतियों के प्रति कम संवेदनशील बनाता है। नीचे एक फ़ॉर्म है जो पांच सेकंड के विलंब से संदेश भेजता है। इस state को कल्पना करें:
- आप “Send” बटन दबाते हैं और “Hello” को Alice को भेजते हैं।
- पांच सेकंड की विलंब समय समाप्त होने से पहले, आप “To” फ़ील्ड के वैल्यू को “Bob” में बदल देते हैं।
आप alert
में क्या प्रदर्शित होने की उम्मीद करते हैं? क्या यह “You said Hello to Alice” दिखाएगा? या क्या यह “You said Hello to Bob” दिखाएगा? अपने ज्ञान के आधार पर एक अनुमान बनाएँ और फिर इसे प्रयास करें:
import { useState } from 'react'; export default function Form() { const [to, setTo] = useState('Alice'); const [message, setMessage] = useState('Hello'); function handleSubmit(e) { e.preventDefault(); setTimeout(() => { alert(`You said ${message} to ${to}`); }, 5000); } return ( <form onSubmit={handleSubmit}> <label> To:{' '} <select value={to} onChange={e => setTo(e.target.value)}> <option value="Alice">Alice</option> <option value="Bob">Bob</option> </select> </label> <textarea placeholder="Message" value={message} onChange={e => setMessage(e.target.value)} /> <button type="submit">Send</button> </form> ); }
React एक रेंडर के इवेंट हैंडलर के भीतर sate वैल्यूस को “निश्चित” रखता है। आपको चिंता करने की आवश्यकता नहीं है कि state कोड चल रहे होने के दौरान बदल गई हो।
लेकिन यदि आप री-रेंडर से पहले नवीनतम state को पढ़ना चाहते हैं तो क्या करें? आपको state अपडेट करने वाली फ़ंक्शन का उपयोग करना चाहिए, जिसका विवरण अगले पेज पर दिया गया है!
Recap
- State को सेट करने से नया रेंडर अनुरोध होता है।
- React आपके कंपोनेंट के बाहर state को एक शेल्फ पर संग्रहीत करता है।
- जब आप
useState
को कॉल करते हैं, तो React आपको उस रेंडर के लिए state का स्नैपशॉट प्रदान करता है। - री-रेंडर करने पर वेरिएबल और इवेंट हैंडलर “बचते” नहीं हैं। प्रत्येक रेंडर में अपने खुद के इवेंट हैंडलर होते हैं।
- प्रत्येक रेंडर (और उसमें स्थित फ़ंक्शन) हमेशा उस स्नैपशॉट को “देखेंगे” जिसे React ने उस रेंडर को दिया है।
- आप इवेंट हैंडलर में state को मानसिक रूप से प्रतिस्थापित कर सकते हैं, ठीक वैसे ही जैसे आप रेंडर किए गए JSX के बारे में सोचते हैं।
- पहले के समय में बनाए गए इवेंट हैंडलर में state वैल्यूस उस रेंडर के होते हैं जिसमें वे बनाए गए थे।
Challenge 1 of 1: Implement a traffic light
यहां एक क्रॉसवॉक लाइट कंपोनेंट है जो बटन दबाने पर टॉगल होता है:
import { useState } from 'react'; export default function TrafficLight() { const [walk, setWalk] = useState(true); function handleClick() { setWalk(!walk); } return ( <> <button onClick={handleClick}> Change to {walk ? 'Stop' : 'Walk'} </button> <h1 style={{ color: walk ? 'darkgreen' : 'darkred' }}> {walk ? 'Walk' : 'Stop'} </h1> </> ); }
इस क्लिक हैंडलर में एक alert जोड़ें। जब लाइट हरी हो और “Walk” कहती हो, तो बटन को क्लिक करने पर “Stop is next” दिखाएँ। जब लाइट लाल हो और “Stop” कहती हो, तो बटन को क्लिक करने पर “Walk is next” दिखाएँ।
क्या यह अंतर करता है कि आप setWalk
कॉल के पहले या उसके बाद alert
डालते हैं?