हिन्दी

डेटा फेचिंग के लिए रिएक्ट सस्पेंस में महारत हासिल करें। लोडिंग स्टेट्स को डिक्लेरेटिव रूप से प्रबंधित करना सीखें, ट्रांज़िशन के साथ यूएक्स में सुधार करें, और एरर बाउंड्रीज़ के साथ त्रुटियों को संभालें।

रिएक्ट सस्पेंस बाउंड्रीज़: डिक्लेरेटिव लोडिंग स्टेट मैनेजमेंट की विस्तृत जानकारी

आधुनिक वेब डेवलपमेंट की दुनिया में, एक सहज और प्रतिक्रियाशील उपयोगकर्ता अनुभव बनाना सर्वोपरि है। डेवलपर्स के सामने सबसे लगातार चुनौतियों में से एक लोडिंग स्टेट्स का प्रबंधन करना है। उपयोगकर्ता प्रोफ़ाइल के लिए डेटा प्राप्त करने से लेकर एप्लिकेशन के एक नए हिस्से को लोड करने तक, प्रतीक्षा के क्षण महत्वपूर्ण होते हैं। ऐतिहासिक रूप से, इसमें isLoading, isFetching, और hasError जैसे बूलियन फ़्लैग्स का एक उलझा हुआ जाल शामिल होता है, जो हमारे कंपोनेंट्स में बिखरे होते हैं। यह अनिवार्य दृष्टिकोण हमारे कोड को अव्यवस्थित करता है, तर्क को जटिल बनाता है, और रेस कंडीशंस जैसे बग्स का एक लगातार स्रोत है।

प्रस्तुत है रिएक्ट सस्पेंस। शुरू में React.lazy() के साथ कोड-स्प्लिटिंग के लिए पेश किया गया, इसकी क्षमताएं रिएक्ट 18 के साथ नाटकीय रूप से विस्तारित हुई हैं और यह एसिंक्रोनस ऑपरेशंस, विशेष रूप से डेटा फेचिंग को संभालने के लिए एक शक्तिशाली, फर्स्ट-क्लास मैकेनिज्म बन गया है। सस्पेंस हमें लोडिंग स्टेट्स को एक डिक्लेरेटिव तरीके से प्रबंधित करने की अनुमति देता है, जिससे हम अपने कंपोनेंट्स को लिखने और उनके बारे में सोचने के तरीके को मौलिक रूप से बदल देते हैं। "क्या मैं लोड हो रहा हूँ?" पूछने के बजाय, हमारे कंपोनेंट बस यह कह सकते हैं, "मुझे रेंडर करने के लिए इस डेटा की आवश्यकता है। जब तक मैं प्रतीक्षा करता हूं, कृपया यह फॉलबैक यूआई दिखाएं।"

यह व्यापक गाइड आपको स्टेट मैनेजमेंट के पारंपरिक तरीकों से लेकर रिएक्ट सस्पेंस के डिक्लेरेटिव पैराडाइम तक की यात्रा पर ले जाएगा। हम यह पता लगाएंगे कि सस्पेंस बाउंड्रीज़ क्या हैं, वे कोड-स्प्लिटिंग और डेटा फेचिंग दोनों के लिए कैसे काम करती हैं, और कैसे जटिल लोडिंग यूआई को व्यवस्थित किया जाए जो आपके उपयोगकर्ताओं को निराश करने के बजाय उन्हें प्रसन्न करे।

पुराना तरीका: मैनुअल लोडिंग स्टेट्स का झंझट

इससे पहले कि हम सस्पेंस की सुंदरता की पूरी तरह से सराहना कर सकें, यह समझना आवश्यक है कि यह किस समस्या का समाधान करता है। आइए एक सामान्य कंपोनेंट को देखें जो useEffect और useState हुक का उपयोग करके डेटा प्राप्त करता है।

कल्पना कीजिए एक ऐसे कंपोनेंट की जिसे उपयोगकर्ता डेटा प्राप्त करने और प्रदर्शित करने की आवश्यकता है:


import React, { useState, useEffect } from 'react';

function UserProfile({ userId }) {
  const [user, setUser] = useState(null);
  const [isLoading, setIsLoading] = useState(true);
  const [error, setError] = useState(null);

  useEffect(() => {
    // Reset state for new userId
    setIsLoading(true);
    setUser(null);
    setError(null);

    const fetchUser = async () => {
      try {
        const response = await fetch(`https://api.example.com/users/${userId}`);
        if (!response.ok) {
          throw new Error('Network response was not ok');
        }
        const data = await response.json();
        setUser(data);
      } catch (err) {
        setError(err);
      } finally {
        setIsLoading(false);
      }
    };

    fetchUser();
  }, [userId]); // Re-fetch when userId changes

  if (isLoading) {
    return <p>Loading profile...</p>;
  }

  if (error) {
    return <p>Error: {error.message}</p>;
  }

  return (
    <div>
      <h1>{user.name}</h1>
      <p>Email: {user.email}</p>
    </div>
  );
}

यह पैटर्न कार्यात्मक है, लेकिन इसके कई नुकसान हैं:

प्रस्तुत है रिएक्ट सस्पेंस: एक आदर्श बदलाव

सस्पेंस इस मॉडल को उलट देता है। कंपोनेंट द्वारा आंतरिक रूप से लोडिंग स्टेट को प्रबंधित करने के बजाय, यह सीधे रिएक्ट को एक एसिंक्रोनस ऑपरेशन पर अपनी निर्भरता बताता है। यदि उसे आवश्यक डेटा अभी तक उपलब्ध नहीं है, तो कंपोनेंट रेंडरिंग को "सस्पेंड" कर देता है।

जब कोई कंपोनेंट सस्पेंड होता है, तो रिएक्ट कंपोनेंट ट्री में सबसे नज़दीकी सस्पेंस बाउंड्री को खोजने के लिए ऊपर जाता है। सस्पेंस बाउंड्री एक कंपोनेंट है जिसे आप अपने ट्री में <Suspense> का उपयोग करके परिभाषित करते हैं। यह बाउंड्री तब तक एक फॉलबैक यूआई (जैसे एक स्पिनर या एक स्केलेटन लोडर) रेंडर करेगी जब तक कि इसके भीतर के सभी कंपोनेंट्स अपनी डेटा निर्भरताओं को हल नहीं कर लेते।

मूल विचार यह है कि डेटा निर्भरता को उस कंपोनेंट के साथ सह-स्थित किया जाए जिसे इसकी आवश्यकता है, जबकि लोडिंग यूआई को कंपोनेंट ट्री में एक उच्च स्तर पर केंद्रीकृत किया जाए। यह कंपोनेंट लॉजिक को साफ करता है और आपको उपयोगकर्ता के लोडिंग अनुभव पर शक्तिशाली नियंत्रण देता है।

एक कंपोनेंट "सस्पेंड" कैसे होता है?

सस्पेंस के पीछे का जादू एक ऐसे पैटर्न में निहित है जो पहली बार में असामान्य लग सकता है: एक प्रॉमिस (Promise) थ्रो करना। एक सस्पेंस-सक्षम डेटा स्रोत इस तरह काम करता है:

  1. जब कोई कंपोनेंट डेटा मांगता है, तो डेटा स्रोत जांचता है कि क्या उसके पास डेटा कैश्ड है।
  2. यदि डेटा उपलब्ध है, तो यह उसे सिंक्रोनस रूप से लौटाता है।
  3. यदि डेटा उपलब्ध नहीं है (यानी, यह वर्तमान में फेच किया जा रहा है), तो डेटा स्रोत उस प्रॉमिस को थ्रो करता है जो चल रहे फेच अनुरोध का प्रतिनिधित्व करता है।

रिएक्ट इस थ्रो किए गए प्रॉमिस को पकड़ लेता है। यह आपके ऐप को क्रैश नहीं करता है। इसके बजाय, यह इसे एक संकेत के रूप में व्याख्या करता है: "यह कंपोनेंट अभी रेंडर करने के लिए तैयार नहीं है। इसे रोकें, और एक फॉलबैक दिखाने के लिए इसके ऊपर एक सस्पेंस बाउंड्री की तलाश करें।" एक बार जब प्रॉमिस हल हो जाता है, तो रिएक्ट कंपोनेंट को फिर से रेंडर करने का प्रयास करेगा, जो अब अपना डेटा प्राप्त करेगा और सफलतापूर्वक रेंडर होगा।

<Suspense> बाउंड्री: आपका लोडिंग यूआई डिक्लेरेटर

<Suspense> कंपोनेंट इस पैटर्न का दिल है। इसका उपयोग करना अविश्वसनीय रूप से सरल है, यह एक एकल, आवश्यक प्रॉप लेता है: fallback


import { Suspense } from 'react';

function App() {
  return (
    <div>
      <h1>My Application</h1>
      <Suspense fallback={<p>Loading content...</p>}>
        <SomeComponentThatFetchesData />
      </Suspense>
    </div>
  );
}

इस उदाहरण में, यदि SomeComponentThatFetchesData सस्पेंड होता है, तो उपयोगकर्ता को "Loading content..." संदेश तब तक दिखाई देगा जब तक डेटा तैयार नहीं हो जाता। फॉलबैक कोई भी मान्य रिएक्ट नोड हो सकता है, एक साधारण स्ट्रिंग से लेकर एक जटिल स्केलेटन कंपोनेंट तक।

क्लासिक उपयोग का मामला: React.lazy() के साथ कोड स्प्लिटिंग

सस्पेंस का सबसे स्थापित उपयोग कोड स्प्लिटिंग के लिए है। यह आपको किसी कंपोनेंट के लिए जावास्क्रिप्ट को लोड करने में देरी करने की अनुमति देता है जब तक कि इसकी वास्तव में आवश्यकता न हो।


import React, { Suspense, lazy } from 'react';

// इस कंपोनेंट का कोड शुरुआती बंडल में नहीं होगा।
const HeavyComponent = lazy(() => import('./HeavyComponent'));

function App() {
  return (
    <div>
      <h2>Some content that loads immediately</h2>
      <Suspense fallback={<div>Loading component...</div>}>
        <HeavyComponent />
      </Suspense>
    </div>
  );
}

यहां, रिएक्ट HeavyComponent के लिए जावास्क्रिप्ट केवल तभी प्राप्त करेगा जब वह पहली बार इसे रेंडर करने का प्रयास करेगा। जब इसे फेच और पार्स किया जा रहा होता है, तो सस्पेंस फॉलबैक प्रदर्शित होता है। यह शुरुआती पेज लोड समय को बेहतर बनाने के लिए एक शक्तिशाली तकनीक है।

आधुनिक फ्रंटियर: सस्पेंस के साथ डेटा फेचिंग

जबकि रिएक्ट सस्पेंस मैकेनिज्म प्रदान करता है, यह एक विशिष्ट डेटा-फेचिंग क्लाइंट प्रदान नहीं करता है। डेटा फेचिंग के लिए सस्पेंस का उपयोग करने के लिए, आपको एक ऐसे डेटा स्रोत की आवश्यकता है जो इसके साथ एकीकृत हो (यानी, एक जो डेटा लंबित होने पर एक प्रॉमिस थ्रो करता है)।

रिले (Relay) और नेक्स्ट.जेएस (Next.js) जैसे फ्रेमवर्क में सस्पेंस के लिए अंतर्निहित, फर्स्ट-क्लास समर्थन है। टैनस्टैक क्वेरी (पूर्व में रिएक्ट क्वेरी) और SWR जैसी लोकप्रिय डेटा-फेचिंग लाइब्रेरी भी इसके लिए प्रायोगिक या पूर्ण समर्थन प्रदान करती हैं।

अवधारणा को समझने के लिए, आइए fetch एपीआई के चारों ओर एक बहुत ही सरल, वैचारिक रैपर बनाएं ताकि इसे सस्पेंस-संगत बनाया जा सके। ध्यान दें: यह शैक्षिक उद्देश्यों के लिए एक सरलीकृत उदाहरण है और उत्पादन-के लिए-तैयार नहीं है। इसमें उचित कैशिंग और त्रुटि प्रबंधन की जटिलताओं का अभाव है।


// data-fetcher.js
// परिणामों को संग्रहीत करने के लिए एक सरल कैश
const cache = new Map();

export function fetchData(url) {
  if (!cache.has(url)) {
    cache.set(url, { status: 'pending', promise: fetchAndCache(url) });
  }

  const record = cache.get(url);

  if (record.status === 'pending') {
    throw record.promise; // यही जादू है!
  }
  if (record.status === 'error') {
    throw record.error;
  }
  if (record.status === 'success') {
    return record.data;
  }
}

async function fetchAndCache(url) {
  try {
    const response = await fetch(url);
    if (!response.ok) {
      throw new Error(`Fetch failed with status ${response.status}`);
    }
    const data = await response.json();
    cache.set(url, { status: 'success', data });
  } catch (e) {
    cache.set(url, { status: 'error', error: e });
  }
}

यह रैपर प्रत्येक URL के लिए एक सरल स्थिति बनाए रखता है। जब fetchData को कॉल किया जाता है, तो यह स्थिति की जाँच करता है। यदि यह लंबित है, तो यह प्रॉमिस थ्रो करता है। यदि यह सफल है, तो यह डेटा लौटाता है। अब, आइए अपने UserProfile कंपोनेंट को इसका उपयोग करके फिर से लिखें।


// UserProfile.js
import React, { Suspense } from 'react';
import { fetchData } from './data-fetcher';

// वह कंपोनेंट जो वास्तव में डेटा का उपयोग करता है
function ProfileDetails({ userId }) {
  // डेटा पढ़ने का प्रयास करें। यदि यह तैयार नहीं है, तो यह सस्पेंड हो जाएगा।
  const user = fetchData(`https://api.example.com/users/${userId}`);

  return (
    <div>
      <h1>{user.name}</h1>
      <p>Email: {user.email}</p>
    </div>
  );
}

// पैरेंट कंपोनेंट जो लोडिंग स्टेट यूआई को परिभाषित करता है
export function UserProfile({ userId }) {
  return (
    <Suspense fallback={<p>Loading profile...</p>}>
      <ProfileDetails userId={userId} />
    </Suspense>
  );
}

अंतर देखें! ProfileDetails कंपोनेंट साफ है और केवल डेटा को रेंडर करने पर केंद्रित है। इसमें कोई isLoading या error स्टेट्स नहीं हैं। यह बस उस डेटा का अनुरोध करता है जिसकी उसे आवश्यकता है। लोडिंग इंडिकेटर दिखाने की जिम्मेदारी पैरेंट कंपोनेंट, UserProfile, पर चली गई है, जो घोषणात्मक रूप से बताता है कि प्रतीक्षा करते समय क्या दिखाना है।

जटिल लोडिंग स्टेट्स को व्यवस्थित करना

सस्पेंस की असली शक्ति तब स्पष्ट होती है जब आप कई एसिंक्रोनस निर्भरताओं के साथ जटिल यूआई बनाते हैं।

एक कंपित यूआई के लिए नेस्टेड सस्पेंस बाउंड्रीज़

आप एक अधिक परिष्कृत लोडिंग अनुभव बनाने के लिए सस्पेंस बाउंड्रीज़ को नेस्ट कर सकते हैं। एक साइडबार, एक मुख्य सामग्री क्षेत्र, और हाल की गतिविधियों की एक सूची के साथ एक डैशबोर्ड पेज की कल्पना करें। इनमें से प्रत्येक को अपने स्वयं के डेटा फेच की आवश्यकता हो सकती है।


function DashboardPage() {
  return (
    <div>
      <h1>Dashboard</h1>
      <div className="layout">
        <Suspense fallback={<p>Loading navigation...</p>}>
          <Sidebar />
        </Suspense>

        <main>
          <Suspense fallback={<ProfileSkeleton />}>
            <MainContent />
          </Suspense>

          <Suspense fallback={<ActivityFeedSkeleton />}>
            <ActivityFeed />
          </Suspense>
        </main>
      </div>
    </div>
  );
}

इस संरचना के साथ:

यह आपको उपयोगकर्ता को यथासंभव शीघ्रता से उपयोगी सामग्री दिखाने की अनुमति देता है, जिससे कथित प्रदर्शन में नाटकीय रूप से सुधार होता है।

यूआई "पॉपकॉर्निंग" से बचना

कभी-कभी, कंपित दृष्टिकोण एक परेशान करने वाले प्रभाव को जन्म दे सकता है जहां कई स्पिनर तेजी से उत्तराधिकार में दिखाई देते हैं और गायब हो जाते हैं, इस प्रभाव को अक्सर "पॉपकॉर्निंग" कहा जाता है। इसे हल करने के लिए, आप सस्पेंस बाउंड्री को ट्री में ऊपर ले जा सकते हैं।


function DashboardPage() {
  return (
    <div>
      <h1>Dashboard</h1>
      <Suspense fallback={<DashboardSkeleton />}>
        <div className="layout">
          <Sidebar />
          <main>
            <MainContent />
            <ActivityFeed />
          </main>
        </div>
      </Suspense>
    </div>
  );
}

इस संस्करण में, एक एकल DashboardSkeleton तब तक दिखाया जाता है जब तक कि सभी चाइल्ड कंपोनेंट्स (Sidebar, MainContent, ActivityFeed) का डेटा तैयार न हो जाए। फिर पूरा डैशबोर्ड एक साथ दिखाई देता है। नेस्टेड बाउंड्रीज़ और एक एकल उच्च-स्तरीय बाउंड्री के बीच का चुनाव एक यूएक्स डिज़ाइन निर्णय है जिसे सस्पेंस लागू करना तुच्छ बना देता है।

एरर बाउंड्रीज़ के साथ त्रुटि प्रबंधन

सस्पेंस एक प्रॉमिस की पेंडिंग स्थिति को संभालता है, लेकिन रिजेक्टेड स्थिति के बारे में क्या? यदि किसी कंपोनेंट द्वारा थ्रो किया गया प्रॉमिस रिजेक्ट हो जाता है (उदाहरण के लिए, एक नेटवर्क त्रुटि), तो इसे रिएक्ट में किसी भी अन्य रेंडरिंग त्रुटि की तरह माना जाएगा।

इसका समाधान एरर बाउंड्रीज़ का उपयोग करना है। एक एरर बाउंड्री एक क्लास कंपोनेंट है जो एक विशेष लाइफसाइकिल मेथड, componentDidCatch() या एक स्टैटिक मेथड getDerivedStateFromError() को परिभाषित करता है। यह अपने चाइल्ड कंपोनेंट ट्री में कहीं भी जावास्क्रिप्ट त्रुटियों को पकड़ता है, उन त्रुटियों को लॉग करता है, और एक फॉलबैक यूआई प्रदर्शित करता है।

यहाँ एक सरल एरर बाउंड्री कंपोनेंट है:


import React from 'react';

class ErrorBoundary extends React.Component {
  constructor(props) {
    super(props);
    this.state = { hasError: false, error: null };
  }

  static getDerivedStateFromError(error) {
    // स्टेट को अपडेट करें ताकि अगला रेंडर फॉलबैक यूआई दिखाए।
    return { hasError: true, error: error };
  }

  componentDidCatch(error, errorInfo) {
    // आप त्रुटि को एक त्रुटि रिपोर्टिंग सेवा में भी लॉग कर सकते हैं
    console.error("Caught an error:", error, errorInfo);
  }

  render() {
    if (this.state.hasError) {
      // आप कोई भी कस्टम फॉलबैक यूआई रेंडर कर सकते हैं
      return <h1>Something went wrong. Please try again.</h1>;
    }

    return this.props.children; 
  }
}

फिर आप तीनों स्थितियों: पेंडिंग, सक्सेस, और एरर को संभालने के लिए एक मजबूत प्रणाली बनाने के लिए एरर बाउंड्रीज़ को सस्पेंस के साथ जोड़ सकते हैं।


import { Suspense } from 'react';
import ErrorBoundary from './ErrorBoundary';
import { UserProfile } from './UserProfile';

function App() {
  return (
    <div>
      <h2>User Information</h2>
      <ErrorBoundary>
        <Suspense fallback={<p>Loading...</p>}>
          <UserProfile userId={123} />
        </Suspense>
      </ErrorBoundary>
    </div>
  );
}

इस पैटर्न के साथ, यदि UserProfile के अंदर डेटा फेच सफल होता है, तो प्रोफ़ाइल दिखाई जाती है। यदि यह पेंडिंग है, तो सस्पेंस फॉलबैक दिखाया जाता है। यदि यह विफल रहता है, तो एरर बाउंड्री का फॉलबैक दिखाया जाता है। तर्क घोषणात्मक, संरचनात्मक और तर्क करने में आसान है।

ट्रांज़िशन: नॉन-ब्लॉकिंग यूआई अपडेट्स की कुंजी

पहेली का एक अंतिम टुकड़ा है। एक उपयोगकर्ता इंटरैक्शन पर विचार करें जो एक नया डेटा फेच ट्रिगर करता है, जैसे कि एक अलग उपयोगकर्ता प्रोफ़ाइल देखने के लिए "अगला" बटन पर क्लिक करना। उपरोक्त सेटअप के साथ, जिस क्षण बटन पर क्लिक किया जाता है और userId प्रॉप बदलता है, UserProfile कंपोनेंट फिर से सस्पेंड हो जाएगा। इसका मतलब है कि वर्तमान में दिखाई देने वाली प्रोफ़ाइल गायब हो जाएगी और लोडिंग फॉलबैक द्वारा प्रतिस्थापित हो जाएगी। यह अचानक और विघटनकारी महसूस हो सकता है।

यहीं पर ट्रांज़िशन आते हैं। ट्रांज़िशन रिएक्ट 18 में एक नई सुविधा है जो आपको कुछ स्टेट अपडेट्स को गैर-जरूरी के रूप में चिह्नित करने देती है। जब एक स्टेट अपडेट को एक ट्रांज़िशन में लपेटा जाता है, तो रिएक्ट पुराने यूआई (पुरानी सामग्री) को प्रदर्शित करना जारी रखेगा, जबकि यह पृष्ठभूमि में नई सामग्री तैयार करता है। यह यूआई अपडेट केवल तभी कमिट करेगा जब नई सामग्री प्रदर्शित होने के लिए तैयार हो।

इसके लिए प्राथमिक एपीआई useTransition हुक है।


import React, { useState, useTransition, Suspense } from 'react';
import { UserProfile } from './UserProfile';

function ProfileSwitcher() {
  const [userId, setUserId] = useState(1);
  const [isPending, startTransition] = useTransition();

  const handleNextClick = () => {
    startTransition(() => {
      setUserId(id => id + 1);
    });
  };

  return (
    <div>
      <button onClick={handleNextClick} disabled={isPending}>
        Next User
      </button>

      {isPending && <span> Loading new profile...</span>}

      <ErrorBoundary>
        <Suspense fallback={<p>Loading initial profile...</p>}>
          <UserProfile userId={userId} />
        </Suspense>
      </ErrorBoundary>
    </div>
  );
}

अब यह होता है:

  1. userId: 1 के लिए प्रारंभिक प्रोफ़ाइल लोड होती है, जिसमें सस्पेंस फॉलबैक दिखाया जाता है।
  2. उपयोगकर्ता "Next User" पर क्लिक करता है।
  3. setUserId कॉल को startTransition में लपेटा गया है।
  4. रिएक्ट मेमोरी में 2 के नए userId के साथ UserProfile को रेंडर करना शुरू कर देता है। इससे यह सस्पेंड हो जाता है।
  5. महत्वपूर्ण रूप से, सस्पेंस फॉलबैक दिखाने के बजाय, रिएक्ट स्क्रीन पर पुराने यूआई (उपयोगकर्ता 1 के लिए प्रोफ़ाइल) को बनाए रखता है।
  6. useTransition द्वारा लौटाया गया isPending बूलियन true हो जाता है, जिससे हमें पुरानी सामग्री को अनमाउंट किए बिना एक सूक्ष्म, इनलाइन लोडिंग इंडिकेटर दिखाने की अनुमति मिलती है।
  7. एक बार जब उपयोगकर्ता 2 के लिए डेटा प्राप्त हो जाता है और UserProfile सफलतापूर्वक रेंडर हो सकता है, तो रिएक्ट अपडेट को कमिट करता है, और नई प्रोफ़ाइल सहजता से दिखाई देती है।

ट्रांज़िशन नियंत्रण की अंतिम परत प्रदान करते हैं, जिससे आप परिष्कृत और उपयोगकर्ता-अनुकूल लोडिंग अनुभव बना सकते हैं जो कभी भी परेशान करने वाले नहीं लगते।

सर्वोत्तम प्रथाएं और वैश्विक विचार

निष्कर्ष

रिएक्ट सस्पेंस केवल एक नई सुविधा से अधिक का प्रतिनिधित्व करता है; यह रिएक्ट अनुप्रयोगों में एसिंक्रोनिसिटी के प्रति हमारे दृष्टिकोण में एक मौलिक विकास है। मैनुअल, अनिवार्य लोडिंग फ़्लैग्स से दूर जाकर और एक घोषणात्मक मॉडल को अपनाकर, हम ऐसे कंपोनेंट लिख सकते हैं जो स्वच्छ, अधिक लचीले और कंपोज करने में आसान हों।

पेंडिंग स्टेट्स के लिए <Suspense>, विफलता स्टेट्स के लिए एरर बाउंड्रीज़, और सहज अपडेट्स के लिए useTransition को मिलाकर, आपके पास एक पूर्ण और शक्तिशाली टूलकिट है। आप सरल लोडिंग स्पिनरों से लेकर जटिल, कंपित डैशबोर्ड खुलासे तक सब कुछ न्यूनतम, अनुमानित कोड के साथ व्यवस्थित कर सकते हैं। जैसे ही आप अपने प्रोजेक्ट्स में सस्पेंस को एकीकृत करना शुरू करेंगे, आप पाएंगे कि यह न केवल आपके एप्लिकेशन के प्रदर्शन और उपयोगकर्ता अनुभव में सुधार करता है, बल्कि आपके स्टेट मैनेजमेंट लॉजिक को भी नाटकीय रूप से सरल बनाता है, जिससे आप उस पर ध्यान केंद्रित कर सकते हैं जो वास्तव में मायने रखता है: बेहतरीन फीचर्स बनाना।