You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Web-Dev-For-Beginners/translations/mr/7-bank-project/4-state-management/README.md

30 KiB

बँकिंग अ‍ॅप तयार करा भाग ४: स्टेट मॅनेजमेंटची संकल्पना

प्री-लेक्चर क्विझ

प्री-लेक्चर क्विझ

परिचय

वेब अ‍ॅप्लिकेशन मोठे होत असताना, डेटा फ्लो ट्रॅक करणे आव्हानात्मक बनते. कोणता कोड डेटा मिळवतो, कोणते पृष्ठ तो वापरते, तो कधी आणि कुठे अपडेट करायचा... हे सर्व व्यवस्थित ठेवणे कठीण होऊ शकते. विशेषतः जेव्हा तुम्हाला तुमच्या अ‍ॅपच्या वेगवेगळ्या पृष्ठांमध्ये डेटा शेअर करायचा असतो, जसे की वापरकर्त्याचा डेटा. स्टेट मॅनेजमेंट ही संकल्पना सर्व प्रकारच्या प्रोग्राम्समध्ये नेहमीच अस्तित्वात असते, परंतु वेब अ‍ॅप्स अधिक जटिल होत असल्याने विकासादरम्यान यावर विचार करणे महत्त्वाचे बनले आहे.

या अंतिम भागात, आपण तयार केलेल्या अ‍ॅपवर पुनर्विचार करू आणि स्टेट कसे व्यवस्थापित केले जाते ते पाहू, ब्राउझर रिफ्रेशला कोणत्याही वेळी समर्थन देणे आणि वापरकर्ता सत्रांमध्ये डेटा टिकवून ठेवणे यासाठी.

पूर्वतयारी

या धड्यासाठी तुम्ही वेब अ‍ॅपचा डेटा फेचिंग भाग पूर्ण केला असावा. तुम्हाला Node.js इंस्टॉल करणे आणि सर्व्हर API स्थानिक पातळीवर चालवणे आवश्यक आहे जेणेकरून तुम्ही खाते डेटा व्यवस्थापित करू शकता.

तुम्ही टर्मिनलमध्ये खालील कमांड चालवून सर्व्हर योग्यरित्या चालत आहे का ते तपासू शकता:

curl http://localhost:5000/api
# -> should return "Bank API v1.0.0" as a result

स्टेट मॅनेजमेंटवर पुनर्विचार करा

मागील धडा मध्ये, आम्ही account नावाच्या ग्लोबल व्हेरिएबलसह अ‍ॅपमध्ये स्टेटची मूलभूत संकल्पना सादर केली होती, ज्यामध्ये सध्या लॉग इन केलेल्या वापरकर्त्याचा बँक डेटा आहे. परंतु, आमच्या सध्याच्या अंमलबजावणीत काही त्रुटी आहेत. डॅशबोर्डवर असताना पृष्ठ रिफ्रेश करून पहा. काय होते?

सध्याच्या कोडमध्ये ३ समस्या आहेत:

  • स्टेट टिकवून ठेवले जात नाही, कारण ब्राउझर रिफ्रेश तुम्हाला लॉगिन पृष्ठावर परत नेते.
  • स्टेट बदलण्यासाठी अनेक फंक्शन्स आहेत. अ‍ॅप मोठा होत असताना, बदल ट्रॅक करणे कठीण होऊ शकते आणि एखादा अपडेट विसरणे सोपे होऊ शकते.
  • स्टेट साफ केला जात नाही, त्यामुळे तुम्ही लॉगआउट क्लिक केल्यावरही खाते डेटा तिथेच राहतो, जरी तुम्ही लॉगिन पृष्ठावर असाल.

आम्ही या समस्यांना एक-एक करून सोडवण्यासाठी आमचा कोड अपडेट करू शकतो, परंतु यामुळे कोड डुप्लिकेशन वाढेल आणि अ‍ॅप अधिक जटिल आणि देखभाल करणे कठीण होईल. किंवा आम्ही काही मिनिटे थांबून आमची रणनीती पुन्हा विचार करू शकतो.

आम्ही येथे कोणत्या समस्यांचे निराकरण करण्याचा प्रयत्न करत आहोत?

स्टेट मॅनेजमेंट म्हणजे या दोन विशिष्ट समस्यांचे निराकरण करण्यासाठी चांगला दृष्टिकोन शोधणे:

  • अ‍ॅपमधील डेटा फ्लो समजण्यास सोपे कसे ठेवायचे?
  • स्टेट डेटा नेहमी वापरकर्ता इंटरफेससह (आणि उलट) समक्रमित कसा ठेवायचा?

एकदा तुम्ही याची काळजी घेतली की, तुम्हाला असलेल्या इतर कोणत्याही समस्यांचे निराकरण आधीच झाले असेल किंवा सोडवणे सोपे झाले असेल. या समस्यांचे निराकरण करण्यासाठी अनेक संभाव्य दृष्टिकोन आहेत, परंतु आम्ही डेटा आणि त्यात बदल करण्याचे मार्ग केंद्रीकृत करणे या सामान्य उपायाचा अवलंब करू. डेटा फ्लो असे असेल:

HTML, वापरकर्ता क्रिया आणि स्टेट यामधील डेटा फ्लो दर्शवणारी योजना

आम्ही येथे डेटा स्वयंचलितपणे दृश्य अपडेट ट्रिगर करतो तो भाग कव्हर करणार नाही, कारण तो Reactive Programming च्या अधिक प्रगत संकल्पनांशी संबंधित आहे. तुम्हाला सखोल अभ्यास करायचा असेल तर हा एक चांगला पुढील विषय आहे.

स्टेट मॅनेजमेंटसाठी वेगवेगळ्या दृष्टिकोनांसह अनेक लायब्ररी उपलब्ध आहेत, Redux हे एक लोकप्रिय पर्याय आहे. मोठ्या वेब अ‍ॅप्समध्ये तुम्हाला कोणत्या संभाव्य समस्यांचा सामना करावा लागू शकतो आणि त्याचे निराकरण कसे करता येईल हे शिकण्यासाठी वापरले जाणारे संकल्पना आणि पॅटर्न पाहा.

कार्य

आम्ही थोडे रिफॅक्टरिंग करून सुरुवात करू. account डिक्लरेशन बदला:

let account = null;

सह:

let state = {
  account: null
};

आमच्या अ‍ॅपमधील सर्व डेटा एका सिंगल स्टेट ऑब्जेक्टमध्ये केंद्रीकृत करण्याची कल्पना आहे. सध्या स्टेटमध्ये फक्त account आहे त्यामुळे फारसा बदल होत नाही, परंतु भविष्यातील सुधारणा करण्याचा मार्ग तयार होतो.

आम्हाला ते वापरणाऱ्या फंक्शन्स देखील अपडेट कराव्या लागतील. register() आणि login() फंक्शन्समध्ये, account = ... बदला state.account = ...;

updateDashboard() फंक्शनच्या शीर्षस्थानी, ही ओळ जोडा:

const account = state.account;

या रिफॅक्टरिंगमुळे फारसा सुधारणा झाली नाही, परंतु पुढील बदलांसाठी पाया तयार करण्याचा उद्देश होता.

डेटा बदल ट्रॅक करा

आता आपण डेटा स्टोअर करण्यासाठी state ऑब्जेक्ट तयार केला आहे, पुढील पायरी म्हणजे अपडेट्स केंद्रीकृत करणे. कोणतेही बदल आणि ते कधी होतात हे ट्रॅक करणे सोपे बनवणे हे उद्दिष्ट आहे.

state ऑब्जेक्टमध्ये बदल होऊ नयेत यासाठी, त्याला immutable मानणे चांगली प्रथा आहे, म्हणजे त्यात अजिबात बदल करता येणार नाही. याचा अर्थ तुम्हाला त्यात काहीही बदल करायचे असल्यास नवीन स्टेट ऑब्जेक्ट तयार करावा लागेल. असे करून, तुम्ही संभाव्य अवांछित side effects पासून संरक्षण तयार करता आणि तुमच्या अ‍ॅपमध्ये नवीन वैशिष्ट्ये लागू करण्याच्या शक्यता उघडता जसे की undo/redo अंमलबजावणी करणे, तसेच डिबग करणे सोपे बनवता. उदाहरणार्थ, तुम्ही स्टेटमध्ये केलेला प्रत्येक बदल लॉग करू शकता आणि बगचा स्रोत समजण्यासाठी बदलांचा इतिहास ठेवू शकता.

JavaScript मध्ये, Object.freeze() वापरून ऑब्जेक्टची immutable आवृत्ती तयार करता येते. जर तुम्ही immutable ऑब्जेक्टमध्ये बदल करण्याचा प्रयत्न केला तर अपवाद निर्माण होईल.

तुम्हाला shallow आणि deep immutable ऑब्जेक्टमधील फरक माहित आहे का? तुम्ही याबद्दल इथे वाचू शकता.

कार्य

नवीन updateState() फंक्शन तयार करूया:

function updateState(property, newData) {
  state = Object.freeze({
    ...state,
    [property]: newData
  });
}

या फंक्शनमध्ये, आम्ही नवीन स्टेट ऑब्जेक्ट तयार करत आहोत आणि spread (...) operator वापरून मागील स्टेटमधून डेटा कॉपी करत आहोत. नंतर आम्ही bracket notation [property] वापरून स्टेट ऑब्जेक्टच्या विशिष्ट प्रॉपर्टीला नवीन डेटासह ओव्हरराइड करतो. शेवटी, आम्ही Object.freeze() वापरून ऑब्जेक्ट लॉक करतो जेणेकरून त्यात बदल होऊ नयेत. सध्या स्टेटमध्ये फक्त account प्रॉपर्टी आहे, परंतु या दृष्टिकोनाने तुम्ही स्टेटमध्ये जितक्या प्रॉपर्टी आवश्यक आहेत तितक्या जोडू शकता.

आम्ही state इनिशियलायझेशन देखील अपडेट करू जेणेकरून प्रारंभिक स्टेट देखील फ्रोझन असेल:

let state = Object.freeze({
  account: null
});

यानंतर, register फंक्शन अपडेट करा आणि state.account = result; असाइनमेंट बदला:

updateState('account', result);

login फंक्शनसह देखील असेच करा, state.account = data; बदला:

updateState('account', data);

आता आपण वापरकर्ता लॉगआउट क्लिक केल्यावर खाते डेटा साफ न होण्याच्या समस्येचे निराकरण करण्याची संधी घेऊ.

नवीन logout() फंक्शन तयार करा:

function logout() {
  updateState('account', null);
  navigate('/login');
}

updateDashboard() मध्ये, return navigate('/login'); रीडायरेक्शन बदला return logout();;

नवीन खाते नोंदणी करा, लॉगआउट करा आणि पुन्हा लॉगिन करा आणि सर्व काही योग्य प्रकारे कार्य करत आहे का ते तपासा.

टिप: तुम्ही updateState() च्या तळाशी console.log(state) जोडून आणि तुमच्या ब्राउझरच्या डेव्हलपमेंट टूल्समध्ये कन्सोल उघडून सर्व स्टेट बदल पाहू शकता.

स्टेट टिकवून ठेवा

बहुतेक वेब अ‍ॅप्सना योग्य प्रकारे कार्य करण्यासाठी डेटा टिकवून ठेवणे आवश्यक असते. सर्व महत्त्वाचा डेटा सामान्यतः डेटाबेसमध्ये संग्रहित केला जातो आणि सर्व्हर API द्वारे प्रवेश केला जातो, जसे की आमच्या बाबतीत वापरकर्ता खाते डेटा. परंतु कधीकधी, ब्राउझरमध्ये चालणाऱ्या क्लायंट अ‍ॅपवर काही डेटा टिकवून ठेवणे देखील मनोरंजक असते, चांगल्या वापरकर्ता अनुभवासाठी किंवा लोडिंग कार्यक्षमता सुधारण्यासाठी.

तुम्हाला तुमच्या ब्राउझरमध्ये डेटा टिकवून ठेवायचा असल्यास, तुम्ही स्वतःला काही महत्त्वाचे प्रश्न विचारले पाहिजेत:

  • डेटा संवेदनशील आहे का? तुम्ही क्लायंटवर कोणताही संवेदनशील डेटा संग्रहित करण्याचे टाळले पाहिजे, जसे की वापरकर्ता पासवर्ड.
  • तुम्हाला किती काळासाठी हा डेटा ठेवायचा आहे? तुम्ही हा डेटा फक्त चालू सत्रासाठी प्रवेश करण्याची योजना आखत आहात की तुम्हाला तो कायमस्वरूपी संग्रहित करायचा आहे?

वेब अ‍ॅपमध्ये माहिती संग्रहित करण्याचे अनेक मार्ग आहेत, तुम्हाला काय साध्य करायचे आहे यावर अवलंबून. उदाहरणार्थ, तुम्ही URL वापरून शोध क्वेरी संग्रहित करू शकता आणि ती वापरकर्त्यांमध्ये शेअर करण्यायोग्य बनवू शकता. तुम्ही HTTP cookies देखील वापरू शकता जर डेटा सर्व्हरसह शेअर करणे आवश्यक असेल, जसे की authentication माहिती.

डेटा संग्रहित करण्यासाठी अनेक ब्राउझर APIs आहेत, त्यापैकी दोन विशेषतः मनोरंजक आहेत:

  • localStorage: एक Key/Value store जो सध्याच्या वेबसाइटसाठी विशिष्ट डेटा वेगवेगळ्या सत्रांमध्ये टिकवून ठेवतो. यात संग्रहित डेटा कधीही एक्सपायर होत नाही.
  • sessionStorage: हे localStorage प्रमाणेच कार्य करते परंतु यात संग्रहित डेटा सत्र संपल्यावर (ब्राउझर बंद झाल्यावर) साफ केला जातो.

लक्षात घ्या की या दोन्ही APIs फक्त strings संग्रहित करण्याची परवानगी देतात. तुम्हाला जटिल ऑब्जेक्ट्स संग्रहित करायचे असल्यास, तुम्हाला JSON.stringify() वापरून ते JSON स्वरूपात सिरीयलाइझ करावे लागेल.

जर तुम्हाला सर्व्हरशिवाय कार्य करणारा वेब अ‍ॅप तयार करायचा असेल, तर IndexedDB API वापरून क्लायंटवर डेटाबेस तयार करणे देखील शक्य आहे. हे प्रगत उपयोग प्रकरणांसाठी किंवा तुम्हाला मोठ्या प्रमाणात डेटा संग्रहित करायचा असल्यास राखीव आहे, कारण ते वापरण्यास अधिक जटिल आहे.

कार्य

आम्हाला आमच्या वापरकर्त्यांना लॉगआउट बटणावर स्पष्टपणे क्लिक करेपर्यंत लॉग इन ठेवायचे आहे, त्यामुळे आम्ही खाते डेटा संग्रहित करण्यासाठी localStorage वापरू. प्रथम, आपण डेटा संग्रहित करण्यासाठी वापरण्यासाठी एक की परिभाषित करूया.

const storageKey = 'savedAccount';

मग updateState() फंक्शनच्या शेवटी ही ओळ जोडा:

localStorage.setItem(storageKey, JSON.stringify(state.account));

यामुळे, वापरकर्ता खाते डेटा टिकवून ठेवला जाईल आणि आपण पूर्वी केंद्रीकृत केलेल्या सर्व स्टेट अपडेट्समुळे नेहमी अद्ययावत राहील. हेच आहे जिथे आपल्याला आपल्या पूर्वीच्या रिफॅक्टरिंगचा फायदा होतो 🙂.

डेटा सेव्ह केला जात असल्याने, अ‍ॅप लोड झाल्यावर तो पुनर्संचयित करण्याची काळजी घेणे आवश्यक आहे. आता आपल्याकडे अधिक इनिशियलायझेशन कोड असणार आहे, त्यामुळे app.js च्या तळाशी असलेल्या आमच्या पूर्वीच्या कोडसह नवीन init फंक्शन तयार करणे चांगली कल्पना असू शकते:

function init() {
  const savedAccount = localStorage.getItem(storageKey);
  if (savedAccount) {
    updateState('account', JSON.parse(savedAccount));
  }

  // Our previous initialization code
  window.onpopstate = () => updateRoute();
  updateRoute();
}

init();

येथे आम्ही सेव्ह केलेला डेटा पुनर्प्राप्त करतो आणि जर काही असेल तर आम्ही त्यानुसार स्टेट अपडेट करतो. हे पृष्ठ अद्यतन दरम्यान स्टेटवर अवलंबून असलेला कोड असू शकतो म्हणून रूट अपडेट करण्यापूर्वी हे करणे महत्त्वाचे आहे.

आम्ही डॅशबोर्ड पृष्ठ आमच्या अ‍ॅप्लिकेशनचे डीफॉल्ट पृष्ठ बनवू शकतो, कारण आता आम्ही खाते डेटा टिकवून ठेवत आहोत. जर कोणताही डेटा सापडला नाही, तर डॅशबोर्ड लॉगिन पृष्ठावर रीडायरेक्ट करण्याची काळजी घेतो. updateRoute() मध्ये, फॉलबॅक return navigate('/login'); बदला return navigate('/dashboard');.

आता अ‍ॅपमध्ये लॉगिन करा आणि पृष्ठ रिफ्रेश करून पहा. तुम्ही डॅशबोर्डवरच राहाल. या अपडेटसह आपण आपल्या सर्व प्रारंभिक समस्यांचे निराकरण केले आहे...

डेटा रिफ्रेश करा

...परंतु आपण नवीन समस्या निर्माण केली असावी. ओह!

test खाते वापरून डॅशबोर्डवर जा, नंतर टर्मिनलवर ही कमांड चालवा आणि नवीन व्यवहार तयार करा:

curl --request POST \
     --header "Content-Type: application/json" \
     --data "{ \"date\": \"2020-07-24\", \"object\": \"Bought book\", \"amount\": -20 }" \
     http://localhost:5000/api/accounts/test/transactions

आता ब्राउझरमध्ये डॅशबोर्ड पृष्ठ रिफ्रेश करून पहा. काय होते? तुम्हाला नवीन व्यवहार दिसतो का?

localStorage मुळे स्टेट अनिश्चित काळासाठी टिकवून ठेवला जातो, परंतु याचा अर्थ तो पुन्हा लॉग आउट आणि लॉग इन केल्याशिवाय कधीही अपडेट होत नाही!

यावर उपाय करण्यासाठी एक संभाव्य रणनीती म्हणजे डॅशबोर्ड लोड झाल्यावर प्रत्येक वेळी खाते डेटा पुन्हा लोड करणे, जेणेकरून जुना डेटा टाळता येईल.

कार्य

नवीन updateAccountData फंक्शन तयार करा:

async function updateAccountData() {
  const account = state.account;
  if (!account) {
    return logout();
  }

  const data = await getAccount(account.user);
  if (data.error) {
    return logout();
  }

  updateState('account', data);
}

हे फंक्शन तपासते की आपण सध्या लॉग इन आहोत आणि नंतर सर्व्हरवरून खाते डेटा पुन्हा लोड करते.

refresh नावाचे आणखी एक फंक्शन तयार करा:

async function refresh() {
  await updateAccountData();
  updateDashboard();
}

हे खाते डेटा अपडेट करते आणि नंतर डॅशबोर्ड पृष्ठाच्या HTML अपडेट करण्याची काळजी घेते. हेच आपल्याला डॅशबोर्ड रूट लोड झाल्यावर कॉल करायचे आहे. रूट डिफिनिशन अपडेट करा:

const routes = {
  '/login': { templateId: 'login' },
  '/dashboard': { templateId: 'dashboard', init: refresh }
};

आता डॅशबोर्ड रिफ्रेश करून पहा, ते अपडेट केलेला खाते डेटा प्रदर्शित करेल.


🚀 आव्हान

आता आपण डॅशबोर्ड लोड झाल्यावर प्रत्येक वेळी खाते डेटा पुन्हा लोड करतो, तुम्हाला असे वाटते का की आपल्याला व्याख्यानानंतरचा प्रश्नमंजूषा

असाइनमेंट

"Add transaction" संवाद लागू करा

असाइनमेंट पूर्ण केल्यानंतरचा एक उदाहरण परिणाम येथे दिला आहे:

"Add transaction" संवादाचे उदाहरण स्क्रीनशॉट दाखवत आहे

अस्वीकरण:
हा दस्तऐवज AI भाषांतर सेवा Co-op Translator वापरून भाषांतरित करण्यात आला आहे. आम्ही अचूकतेसाठी प्रयत्नशील असलो तरी कृपया लक्षात ठेवा की स्वयंचलित भाषांतरांमध्ये त्रुटी किंवा अचूकतेचा अभाव असू शकतो. मूळ भाषेतील दस्तऐवज हा अधिकृत स्रोत मानला जावा. महत्त्वाच्या माहितीसाठी व्यावसायिक मानवी भाषांतराची शिफारस केली जाते. या भाषांतराचा वापर करून उद्भवलेल्या कोणत्याही गैरसमज किंवा चुकीच्या अर्थासाठी आम्ही जबाबदार राहणार नाही.