ভাই, খুব সুন্দর প্রশ্ন করেছেন — ধন্যবাদ! 🙏
আমি ইংরেজিতে লিখেছি কারণ অনেক সময় আপনি ইংরেজিতে লিখে থাকেন, আর React documentation বা কোডগুলাও ইংরেজিতেই থাকে। কিন্তু এখন যেহেতু আপনি মনে করিয়ে দিলেন, আমি আপনার বইয়ের জন্য Bangla-ভাষায় সহজ করে পুরো Chapter লিখে দিবো — যেন একদম নতুনদের জন্য বোধগম্য হয়।
চলুন তাহলে আপনার বইয়ের এই Chapter-এর Section 1 (Intro) এবং Section 2 (Effect Lifecycle Summary) বাংলায় লিখে দিই:
অধ্যায়: Reactive Effect-এর Lifecycle
Section 1: ভূমিকা
React-এ প্রতিটি component-এর একটি জীবনচক্র থাকে — যেমন:
- Mount হয়: যখন component প্রথমবার স্ক্রিনে আসে
- Update হয়: যখন props বা state পরিবর্তন হয়
- Unmount হয়: যখন component স্ক্রিন থেকে চলে যায়
কিন্তু useEffect
হুক একদম আলাদা। এটি component-এর মতো তিনটি ধাপে কাজ করে না।
useEffect
কেবল দুইটা কাজ করে:
- কোনো কিছু synchronize করা শুরু করে (যেমন: সার্ভারের সাথে কানেক্ট হওয়া)
- পরে তা বন্ধ করে দেয় (যেমন: disconnect করা)
এই synchronize করার কাজটি একাধিকবারও হতে পারে — যদি আপনার props বা state পরিবর্তন হয়।
React নিজেই আপনাকে সতর্ক করে দেয় যাতে আপনি useEffect
-এর dependency গুলো ঠিকভাবে দেন। এটার জন্য React-এর একটি linter rule আছে।
আপনি শিখবেন:
✅ component lifecycle আর effect lifecycle-এর পার্থক্য ✅ কিভাবে একেকটা Effect আলাদাভাবে ভাবতে হয় ✅ কখন Effect আবার চালু হবে আর কেন ✅ dependency array কিভাবে কাজ করে ✅ reactive মানে কী ✅ খালি dependency array দিলে কী হয় ✅ React কিভাবে dependency ঠিক আছে কিনা যাচাই করে ✅ আপনি যদি linter-এর সাথে একমত না হন তখন কী করবেন
Section 2: Effect-এর লাইফসাইকেল কেমন?
প্রতিটি React component নিচের মতো করে চলে:
- স্ক্রিনে mount হয়
- পরে update হয় (যখন props বা state পরিবর্তন হয়)
- শেষে unmount হয় (স্ক্রিন থেকে চলে যায়)
এভাবে component-এর lifecycle বোঝা যায়।
কিন্তু useEffect
এর জন্য এইভাবে ভাবা ভুল। useEffect
component lifecycle-এর উপর নির্ভর করে না। বরং useEffect
কেবল বলে:
“আমি কীভাবে বাইরের কোনো জিনিসের সাথে sync করবো, এবং কখন সেটাকে বন্ধ করবো।”
এই syncing অনেকবার ঘটতে পারে — প্রতিবার যখন useEffect
এর dependency (যেমন props/state) পরিবর্তন হয়।
উদাহরণ: Chat Server এ কানেক্ট করা
const serverUrl = "https://localhost:1234";
function ChatRoom({ roomId }) {
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => {
connection.disconnect();
};
}, [roomId]);
}
এখানে useEffect
এর ভিতরে কী ঘটছে?
✅ Synchronize শুরু হচ্ছে:
const connection = createConnection(serverUrl, roomId);
connection.connect();
✅ Synchronize বন্ধ করা হচ্ছে:
return () => {
connection.disconnect();
};
আপনি হয়তো ভাবছেন — component যখন mount হয়, তখনই connect হবে আর unmount হলে disconnect হবে।
কিন্তু না! যদি roomId
props পরিবর্তন হয়, তখনও Effect আবার চলবে।
React তখন কী করবে?
- আগের room (
"general"
) থেকে disconnect করবে - নতুন room (
"travel"
) এ connect করবে
এই পুরো কাজটাই React করে দিবে — কারণ আপনি useEffect
-এর মাধ্যমে বলে দিয়েছেন:
- কীভাবে connect করতে হয়
- কীভাবে disconnect করতে হয়
📝 নোট:
সব useEffect
-এ cleanup দরকার হয় না।
তবে যদি আপনি কোনো টাইমার, সাবস্ক্রিপশন, সার্ভার কানেকশন ইত্যাদির মতো কাজ করেন — তাহলে অবশ্যই cleanup function ব্যবহার করা উচিত।
আমরা দেখবো কীভাবে useEffect
দিয়ে কোনো ইভেন্টে সাবস্ক্রাইব করা যায় এবং কম্পোনেন্ট unmount হলে সেই সাবস্ক্রিপশন কীভাবে বন্ধ করা হয়।
✅ উদাহরণ: উইন্ডোর রিসাইজ ইভেন্টে সাবস্ক্রাইব
import { useEffect, useState } from "react";
function WindowSizeTracker() {
const [width, setWidth] = useState(window.innerWidth);
useEffect(() => {
// সাবস্ক্রাইব করছি: রিসাইজ হলে নতুন width সেট করবো
const handleResize = () => {
setWidth(window.innerWidth);
};
window.addEventListener("resize", handleResize);
// 🧹 ক্লিনআপ: কম্পোনেন্ট unmount হলে সাবস্ক্রিপশন বন্ধ
return () => {
window.removeEventListener("resize", handleResize);
};
}, []); // খালি dependency মানে শুধু একবার রান করবে
return (
<div>
<h2>বর্তমান উইন্ডো প্রস্থ: {width}px</h2>
</div>
);
}
export default WindowSizeTracker;
🧠 ব্যাখ্যা বাংলায়:
ধাপ | ব্যাখ্যা |
---|---|
useEffect(() => {...}, []) | কম্পোনেন্ট লোড হওয়ার পর শুধুমাত্র একবার চলবে। |
window.addEventListener | ব্রাউজারে কোনো ইভেন্ট শুনে — এখানে রিসাইজ ইভেন্টে লিসেনার বসানো হলো। |
return () => {...} | এই অংশটা useEffect এর ক্লিনআপ ফাংশন — সাবস্ক্রিপশন বন্ধ করার জন্য দরকার, নয়তো মেমোরি লিক হতে পারে। |
setWidth(window.innerWidth) | যখন ইউজার উইন্ডোর সাইজ পরিবর্তন করে, তখন নতুন প্রস্থ state-এ সেট হয়। |
💡 মনে রাখার মতো বিষয়
- সাবস্ক্রিপশন মানে হচ্ছে এমন কিছু শোনা বা ট্র্যাক করা — যেমন রিসাইজ, স্ক্রল, ওয়েবসকেট, টাইমার ইত্যাদি।
- ক্লিনআপ ফাংশন না দিলে পুরনো ইভেন্ট লিসেনার থেকে সমস্যা হতে পারে।
অবশ্যই ভাই, নিচের অংশে Section 2.2 বাংলায় সহজ ভাষায় লিখে দিলাম, আগের মতো format ধরে রেখে:
Section 2.2: React কিভাবে আপনার Effect আবার synchronize করে
আপনার ChatRoom
component এর কথা মনে আছে তো? ধরুন, এটা আগেও "general"
নামে একটা room পেয়েছিল roomId
হিসেবে। এখন সেটা পরিবর্তন হয়ে "travel"
হয়েছে।
এখন React কী করবে?
React বুঝবে, Effect-টি আবার synchronize করতে হবে, কারণ আপনি অন্য একটার সাথে সংযোগ করতে চাইছেন।
✅ Step 1: পুরোনো Synchronization বন্ধ করা
React প্রথমে পুরোনো Effect এর cleanup function কল করবে। এই function আগের "general"
room এর সাথে সংযোগ বন্ধ করার কাজ করে।
function ChatRoom({ roomId /* "general" */ }) {
useEffect(() => {
const connection = createConnection(serverUrl, roomId); // "general" রুমে সংযুক্ত হচ্ছে
connection.connect();
return () => {
connection.disconnect(); // "general" রুম থেকে সংযোগ বিচ্ছিন্ন করছে
};
}, [roomId]);
}
✅ Step 2: নতুন Synchronization শুরু করা
এরপর React নতুন roomId
মান সহ নতুন Effect চালাবে। এবার roomId
হচ্ছে "travel"
।
function ChatRoom({ roomId /* "travel" */ }) {
useEffect(() => {
const connection = createConnection(serverUrl, roomId); // "travel" রুমে সংযুক্ত হচ্ছে
connection.connect();
// ...
}, [roomId]);
}
এভাবে React আপনার roomId
-এর উপর নির্ভর করে পুরাতন সংযোগ বন্ধ করে, আর নতুন সংযোগ চালু করে — যেটা ব্যবহারকারী UI-তে বেছে নিয়েছে।
এটার জন্য আপনি নিজে কিছু করতে হয় না — শুধু dependency ঠিক রাখলেই React সব ম্যানেজ করে দেয়।
🎯 প্রতি বার roomId পরিবর্তন হলে কী হয়?
ধরুন, ব্যবহারকারী আবার roomId
পরিবর্তন করে "travel"
থেকে "music"
এ চলে গেল। তখন React আবার:
- আগের Effect এর cleanup function কল করে –
"travel"
থেকে disconnect হয় - নতুন Effect চালায় –
"music"
room-এ connect হয়
এভাবেই প্রতিবার roomId
পরিবর্তন হলে Effect re-synchronize হয়।
🧹 Unmount হলে কী হয়?
যখন ব্যবহারকারী অন্য কোনো স্ক্রিনে চলে যায়, তখন ChatRoom
component unmount হয়ে যায়। তখন React শেষবারের মতো cleanup function চালায় — অর্থাৎ "music"
রুম থেকেও disconnect করে দেয়।
📌 মূল কথা:
React প্রতিবার নতুন props (যেমন roomId
) পেলে:
- আগের effect বন্ধ করে (cleanup)
- নতুন effect চালায় (sync শুরু করে)
এভাবেই React আপনার useEffect
-কে বারবার re-synchronize করে — আপনার দেওয়া dependency ([roomId]
) দেখে দেখে।
অবশ্যই ভাই, নিচে Section 2.3 বাংলায় সহজভাবে এবং আগের ধারা অনুযায়ী অনুবাদ করে দিলাম:
Section 2.3: Effect-এর দৃষ্টিকোণ থেকে চিন্তা করা
আমরা এখন পর্যন্ত যা শিখেছি, একবার ChatRoom
component-এর দৃষ্টিকোণ থেকে পুরো বিষয়টা সংক্ষেপে দেখি:
ChatRoom
প্রথমবার তৈরি (mount) হলো, তখনroomId
ছিল"general"
- তারপর
roomId
পরিবর্তন হলো"travel"
এ - আবার
roomId
পরিবর্তন হলো"music"
এ - শেষে
ChatRoom
component unmount হলো (মানে, UI থেকে গায়েব হলো)
এই সময়গুলোতে আপনার useEffect
যা যা করেছিল:
"general"
room-এ সংযুক্ত হয়েছিল"general"
থেকে সংযোগ বিচ্ছিন্ন করে"travel"
room-এ সংযুক্ত হয়েছিল"travel"
থেকে সংযোগ বিচ্ছিন্ন করে"music"
room-এ সংযুক্ত হয়েছিল"music"
room থেকেও সংযোগ বিচ্ছিন্ন করে দিয়েছিল
🤔 এবার চিন্তা করি Effect-এর দৃষ্টিকোণ থেকে
useEffect(() => {
// roomId অনুযায়ী নতুন সংযোগ তৈরি হয়
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => {
// যখন effect বন্ধ হয়, তখন সংযোগ বিচ্ছিন্ন হয়
connection.disconnect();
};
}, [roomId]);
এই কোডটা দেখলে মনে হবে, যেন effect-টা একটা নির্দিষ্ট সময়ের জন্য শুরু হয় এবং পরে থামে — একবারে একটাই room handle করে।
এইভাবে চিন্তা করলে সহজ হবে:
- প্রথমে Effect
"general"
room-এ connect করল, যতক্ষণ না সেটা disconnect হলো - এরপর
"travel"
room-এ connect করল, যতক্ষণ না সেটা disconnect হলো - এরপর
"music"
room-এ connect করল, যতক্ষণ না সেটা disconnect হলো
⚠️ Component-এর দৃষ্টিকোণ থেকে চিন্তা করলে জটিলতা বাড়ে
আগে আমরা component-এর lifecycle অনুযায়ী ভাবছিলাম — render হওয়ার পরে effect চলে, unmount হওয়ার আগে cleanup হয়। এইভাবে চিন্তা করলে দ্রুত জটিল হয়ে যায়, কারণ তখন component কখন mount, কখন update, কখন unmount — সব ভাবতে হয়।
✅ বরং Effect-এর "start/stop" চক্র নিয়ে ভাবুন
আপনার চিন্তা এমন হওয়া উচিত:
এই Effect শুরু হলে কী করতে হবে? এই Effect বন্ধ হলে (cleanup হলে) কী করতে হবে?
আর কিছুই ভাবার দরকার নেই। Component কখন mount হচ্ছে বা update হচ্ছে, সেটা মাথায় না রাখলেও চলবে।
যদি আপনি Effect-কে এমনভাবে লেখেন, যেন সেটা যতবার দরকার হয় ততবার শুরু ও বন্ধ হতে পারে — তাহলে আপনার code দৃঢ় (resilient) হবে। কোনো bug হবে না।
🎨 একদম যেমন JSX লিখি
যখন আপনি JSX লেখেন, তখন ভাবেন না component "mount" হচ্ছে নাকি "update" — আপনি শুধু বলেন "স্ক্রিনে এটা দেখাও", আর React বাকিটা ম্যানেজ করে।
একইভাবে, Effect লেখার সময় আপনি শুধু বলেন:
- কীভাবে শুরু হবে (connect)
- কীভাবে বন্ধ হবে (disconnect)
React সব বাকিটা smartly handle করে।
অবশ্যই ভাই, নিচে Section 2.4: "How React verifies that your Effect can re-synchronize" অংশটিকে বাংলায় সহজ ভাষায় ব্যাখ্যা করে দিলাম:
Section 2.4: React কীভাবে যাচাই করে যে আপনার Effect পুনরায় সঠিকভাবে sync করতে পারবে
🧪 লাইভ উদাহরণ বিশ্লেষণ
উপরের কোডটি একটা simple চ্যাট অ্যাপের ডেমো, যেখানে আপনি:
- কোন রুমে চ্যাট করবেন তা সিলেক্ট করতে পারেন (
roomId
) - আর চ্যাট চালু/বন্ধ (mount/unmount) করতে পারেন একটি বাটনের মাধ্যমে
🔍 যখন আপনি প্রথমবার "Open chat" চাপ দেন, React ChatRoom
component কে mount করে। তখন আপনি কনসোলে নিচের ৩টি লগ দেখতে পারেন:
- ✅
Connecting to "general" room at https://localhost:1234...
- ❌
Disconnected from "general" room at https://localhost:1234.
- ✅
Connecting to "general" room at https://localhost:1234...
❗এই ৩টা লগের মানে কী?
- প্রথম connect এবং তারপরই disconnect হওয়া শুধুমাত্র development mode-এ হয়।
- React ইচ্ছা করে component-কে দুইবার mount করে এটা যাচাই করার জন্য যে আপনি আপনার
useEffect
-এর cleanup logic ঠিকভাবে লিখেছেন কি না।
এটা অনেকটা দরজা খুলে আবার বন্ধ করে দেখা — যে দরজার লকটা ঠিকভাবে কাজ করছে কিনা। React আপনার effect-কে জোর করে একবার বন্ধ করে আবার চালায়, যেন দেখে আপনার cleanup ঠিকঠাক কাজ করছে কি না।
✅ "Effect পুনরায় sync হতে পারে" – এর মানে?
"Effect পুনরায় synchronize" মানে:
- আগের কিছু বন্ধ করে দেওয়া (cleanup)
- তারপর নতুনভাবে আবার শুরু করা (connect)
React এটা করে যখন:
roomId
বদলায় (যেমন general → travel → music)serverUrl
বদলায়- বা component কে remount করতে হয় (যেমন চ্যাট ক্লোজ করে আবার ওপেন করলে)
⚙️ রি-Sync কেন গুরুত্বপূর্ণ?
আপনার Effect-এর কাজ যদি একটা বাহ্যিক সংযোগ বা resource-এর সাথে যুক্ত হয় (যেমন: API call, WebSocket, chat room connection) — তাহলে সেই সংযোগ ঠিকভাবে বন্ধ করে নতুন করে শুরু করা জরুরি।
React চায় আপনি এরকম সংবেদনশীল কোডে ভুল না করেন, তাই ডেভেলপমেন্ট মোডে দুইবার চালিয়ে দেখে নেয় সব ঠিক আছে কিনা।
🧠 মনে রাখবেন:
- আপনি যদি আপনার
useEffect
-এর ভিতরে ভালোভাবেreturn () => {}
দিয়ে cleanup দেন, তাহলে এই "double run" কোনো সমস্যা করবে না। - Production mode-এ React একবারই চালাবে। এই double-run শুধু dev মোডে testing/checking এর জন্য।
📌 উপসংহার:
React নিশ্চিত হতে চায়:
"আপনার Effect ঠিকভাবে বন্ধ হয় এবং আবার চালু হতে পারে — সেটা
roomId
বদলানোর সময়ই হোক, বা dev মোডে React নিজে ইচ্ছা করে সেটা restart করুক।"
এই কারণেই useEffect
লেখার সময় ভালোভাবে "start" এবং "stop" behavior handle করা শেখা জরুরি।
অবশ্যই ভাই, নিচে Section 2.5: "How React knows that it needs to re-synchronize the Effect" অংশটিকে বাংলায় সহজ ভাষায় ব্যাখ্যা করলাম:
Section 2.5: React কীভাবে বোঝে যে Effect-কে পুনরায় চালানো (re-synchronize) দরকার
❓প্রশ্ন: roomId
বদলালে React কীভাবে বুঝলো Effect আবার চালাতে হবে?
উত্তর:
কারণ আপনিই React-কে বলে দিয়েছেন যে এই Effect-এর কাজটা roomId
-এর উপর নির্ভর করে। আপনি useEffect(..., [roomId])
-এ dependency array তে roomId
যুক্ত করেছেন।
function ChatRoom({ roomId }) {
// roomId প্রপটি সময়ের সাথে বদলাতে পারে
useEffect(() => {
const connection = createConnection(serverUrl, roomId); // এই Effect-এর ভিতরে roomId ব্যবহার হচ্ছে
connection.connect();
return () => {
connection.disconnect(); // পরিষ্কারভাবে disconnect করার logic
};
}, [roomId]); // তাই dependency হিসেবে roomId দেওয়া হয়েছে
}
⚙️ এটি কীভাবে কাজ করে?
- আপনি জানেন
roomId
একটা prop — অর্থাৎ এটি সময়ের সাথে বদলাতে পারে। - আপনি
useEffect
-এর ভিতরেroomId
ব্যবহার করছেন — কাজেই এই EffectroomId
-এর উপরে নির্ভর করছে। - তাই আপনি dependency array-এ
[roomId]
দিয়েছেন, যেন React বুঝতে পারেroomId
বদলালে Effect-টি আবার চালাতে হবে।
🔁 React কীভাবে পরিবর্তন চেক করে?
- প্রতিবার যখন আপনার component re-render হয়, React চেক করে আপনি
useEffect(..., [roomId])
এ যে value গুলো দিয়েছেন সেগুলোর মধ্যে কোনোটা বদলেছে কিনা। - React এই তুলনা করে
Object.is()
দিয়ে।
উদাহরণ:
- প্রথমবার আপনি
[roomId]
=["general"]
দিলেন - পরেরবার আপনি
[roomId]
=["travel"]
দিলেন
React তুলনা করবে:
Object.is("general", "travel"); // false
তাই React বুঝবে value বদলেছে → Effect কে re-run করবে।
কিন্তু যদি মান না বদলায় (যেমন আবার "general"
হয়), তাহলে Effect চলবে না — আগের অবস্থায়ই থাকবে।
🧠 মনে রাখবেন:
Dependency array ঠিকঠাক না দিলে হয়তো Effect পুরনো ডেটা নিয়ে কাজ করবে বা কখনোই চলবে না।
✅ সংক্ষেপে:
ঘটনা | React কী করে? |
---|---|
dependency array-এ থাকা মান বদলায় | Effect cleanup করে আবার চালায় |
মান বদলায় না | আগের Effect চলতে থাকে |
dependency না দিলে | React বোঝে না কখন চালাতে হবে — Bug হতে পারে |
আপনি যদি চান আমি এটা নিয়ে রিয়েল লাইফ উদাহরণসহ আরও ব্যাখ্যা করি (যেমন: API call, socket connection, authentication token change ইত্যাদি), বললেই করে দেব ভাই। ✨
Section 2.5: প্রতিটি Effect মানে একটি আলাদা সিঙ্ক্রোনাইজেশন প্রসেস
🚫 ভুল ধারণা: সবকিছু একসাথে একটাই Effect-এ লেখা উচিত
অনেক সময় আমরা দেখি কোনো কাজ একসাথে "একই সময়ে" হয়, তাই ভেবে নিই সেগুলোকে একই useEffect
-এ লিখে ফেললেই ভালো।
ধরুন, আপনি চান:
- ইউজার যখন নতুন chat room-এ যায়, তখন analytics-এ log করা হোক
- একই সময়ে, আপনি server-এর সাথে connection করাচ্ছেন
তখন হয়তো এমন কোড লিখবেন:
useEffect(() => {
logVisit(roomId); // log করছে
const connection = createConnection(serverUrl, roomId); // connect করছে
connection.connect();
return () => connection.disconnect();
}, [ServerUrl, roomId]);
❌ কিন্তু এটা ঠিক নয়। কারণ: এখানে দুটো ভিন্ন কাজ- সার্ভার চেঞ্জ হলেও আমরা দেখব, ইউজার জেনারেল রুমে ভিজিট করছেন। যা ঠিক নয়, ইউজার অলরেডি আছেন।
(analytics log + server connect) একসাথে লিখে ফেলেছেন। ভবিষ্যতে যদি আপনি অন্য কিছুর কারণে এই Effect-এ নতুন dependency যোগ করেন, তখন logVisit()
অপ্রয়োজনে আবার চালু হয়ে যেতে পারে — যেটা আপনি চান না।
✅ সঠিক উপায়: Effect-কে ভাগ করে নেওয়া
যেহেতু logVisit(roomId)
এবং createConnection(...)
দুটি আলাদা প্রসেস, তাই প্রতিটাকে আলাদা useEffect
-এ লিখুন:
useEffect(() => {
logVisit(roomId); // শুধু logVisit এর জন্য আলাদা Effect
}, [roomId]);
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect(); // শুধু connection এর জন্য আলাদা Effect
}, [ServerUrl, roomId]);
🔑 মূল পয়েন্ট:
প্রতিটি
useEffect
হচ্ছে একেকটা আলাদা কাজের (synchronization process) controller।
ভালো চিহ্ন কী? যদি আপনি একটি Effect ডিলিট করেন, অন্য Effect ঠিকঠাক কাজ করে — তাহলে বুঝবেন ওরা একে অপরের উপর নির্ভর করে না, তাই আলাদা লেখা উচিত।
📌 মনে রাখার নিয়ম:
ভুল | ঠিক |
---|---|
একাধিক কাজ এক Effect-এ লেখা | প্রতিটি কাজ আলাদা Effect-এ |
শুধু "কোড দেখতে ছোট" তাই আলাদা করা | বুঝে আলাদা করা — কাজ আলাদা কিনা দেখে |
এক Effect-এ সব কাজ গুঁজে ফেলা | প্রতিটি Effect হচ্ছে একেকটা দায়িত্বশীল লোক |
🎯 বাংলায় সংক্ষেপে:
"একজন মানুষ একসাথে চা বানাবে, আবার অফিসের attendance-ও রেজিস্টারে লিখবে — এটা ঠিক না। চা বানানো আর attendance রেজিস্টার করা, দুইজন আলাদা লোক দিলেই কাজ গুছানো হয়।"
ঠিক React-এও তাই — প্রতিটি Effect আলাদা দায়িত্বে থাকুক।
আপনি চাইলে আমি এই উদাহরণটা বাস্তব app context (যেমন Firebase auth, notifications, বা API fetch) দিয়েও বুঝিয়ে দিতে পারি ভাই। বললেই করে দেব ❤️
📘 Section 3: Effect "react" করে কাদের উপর?
🔍 সমস্যা:
const serverUrl = "https://localhost:1234";
এই ভ্যালু useEffect
-এর ভিতরে ব্যবহৃত হচ্ছে, কিন্তু useEffect
-এর ডিপেন্ডেন্সি অ্যারেতে এটা নেই:
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, [roomId]); // serverUrl নেই!
❓ তাহলে serverUrl
কেন ডিপেন্ডেন্সি নয়?
কারণ serverUrl
হচ্ছে একটা non-reactive value। এটা কম্পোনেন্ট রেন্ডার হবার সময় বদলায় না। তাই এটা useEffect
এর ডিপেন্ডেন্সি অ্যারেতে রাখার দরকার নেই।
⚠️ কিন্তু roomId
কি reactive?
হ্যাঁ! roomId
হলো prop — এটি প্রতি রেন্ডারে বদলাতে পারে। তাই এটি হচ্ছে reactive value, এবং তাই এটিকে [roomId]
ডিপেন্ডেন্সি অ্যারেতে রাখতে হয়।
🧪 যদি serverUrl
হয় state?
const [serverUrl, setServerUrl] = useState("https://localhost:1234");
তাহলে এটি একটি reactive value হয়ে যায়। মানে, ইউজার যদি input দিয়ে এটাকে বদলায়, তাহলে useEffect
-কে আবার চালাতে হবে নতুন serverUrl
অনুযায়ী।
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, [roomId, serverUrl]); // এবার দুটোই reactive, তাই দুটোই ডিপেন্ডেন্সি
🧠 মনে রাখো:
মান কোথা থেকে আসে? | reactive কিনা? | useEffect এ dependency হবে? |
---|---|---|
প্রপস (props) | ✅ হ্যাঁ | ✅ হ্যাঁ |
স্টেট (state) | ✅ হ্যাঁ | ✅ হ্যাঁ |
ফাংশনের বাইরে ডিফাইন করা | ❌ না | ❌ না (যদি না চেঞ্জ হয়) |
🎓 উপসংহার (Section 3.1 hint):
"reactive value" বলতে বোঝায় যেসব ভ্যালু কম্পোনেন্ট রেন্ডার করার সময় পরিবর্তিত হতে পারে — যেমন: props, state, বা রেন্ডার ফাংশনের ভিতরে ডিক্লেয়ার করা ভ্যালু।
অবশ্যই ভাই, নিচের অংশটা আমি সহজ বাংলায় ব্যাখ্যা করছি — যেন আপনি React-এর useEffect([])
এর ব্যবহার ভালোভাবে বুঝতে পারেন।
🧠 Section 3.1: খালি []
dependencies কী বোঝায়?
🔍 উদাহরণ কোড:
const serverUrl = "https://localhost:1234";
const roomId = "general";
function ChatRoom() {
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, []); // ✅ খালি dependency মানে: শুধু একবার চলবে
return <h1>Welcome to the {roomId} room!</h1>;
}
🟢 যখন useEffect([])
লেখা হয়, তখন কী হয়?
👉 আপনি যখন useEffect
-এ খালি []
দেন, তখন এর মানে হচ্ছে:
- এই Effect শুধু একবার চালু হবে, যখন component প্রথমবার রেন্ডার হবে (mounting সময়)।
- এবং component বন্ধ হলে (unmount) তখন cleanup function চালাবে — মানে disconnect করবে।
🔁 এই Effect আবার চালু হবে না যতক্ষণ না আপনি component টা destroy করে আবার render করেন।
🔗 বাস্তব উদাহরণ দিয়ে বোঝাই:
আপনারা হয়তো Firebase, socket.io, অথবা কোনো API-র সাথে connection করেন।
useEffect(() => {
connectToServer(); // ✅ শুধু প্রথমবার সংযোগ
return () => disconnectFromServer(); // ❌ যখন বন্ধ হবে তখন ডিসকানেক্ট
}, []);
এই pattern মানে: "আমি একবার সংযোগ করব, আর একবার ডিসকানেক্ট করব" — এবং সেটা component এর জীবনচক্রের শুরু আর শেষে হবে।
🧠 তাহলে কেন [roomId]
না দিয়ে []
দিলাম?
কারণ এই উদাহরণে:
const serverUrl = "https://localhost:1234";
const roomId = "general";
👉 এরা reactive না — মানে props
, state
, বা context থেকে আসেনি। এগুলো component এর বাইরে declare করা — তাই পরিবর্তন হবে না।
এই কারণে useEffect([])
দিয়েই safe — কারণ কোনো reactive মানে depend করছে না।
🧩 কিন্তু ভবিষ্যতে যদি roomId বা serverUrl dynamic হয়?
ধরুন আপনি ইউজারকে allow করলেন roomId বদলাতে:
function ChatRoom({ roomId }) {
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, [roomId]); // ✅ এখন এটা reactive তাই dependency যোগ করলাম
}
এখন আপনার কোডের মূল কাজ একই থাকলো, শুধু dependency array-এ roomId
যুক্ত করলেন।
✅ মূল পয়েন্ট বাংলায়:
বিষয় | অর্থ |
---|---|
useEffect([]) | শুধুমাত্র একবার চালাবে যখন component তৈরি হয় |
dependency না দিলে | React বুঝবে না কবে effect আবার চালানো দরকার |
future-proof | যদি পরের বার roomId বা serverUrl reactive হয়, তখন শুধু dependency array-এ যুক্ত করলেই চলবে |
নিচে Section 3.2: কম্পোনেন্ট বডিতে ডিক্লেয়ার করা সব ভ্যারিয়েবলই reactive হয় — এই অংশটিকে আমি সহজ বাংলায় ব্যাখ্যা করছি, যেন তোমার React বইয়ের বাংলা সংস্করণে এটি শিক্ষার্থীদের কাছে পরিষ্কারভাবে পৌঁছায়।
📘 Section 3.2: কম্পোনেন্ট বডিতে ডিক্লেয়ার করা সব ভ্যারিয়েবলই Reactive
✅ মূল ধারণা:
props
আর state
ছাড়া আরও অনেক কিছুই reactive হতে পারে।
যেকোনো ভ্যারিয়েবল যা তুমি কম্পোনেন্টের ভিতরে তৈরি করো এবং যেটি props
বা state
এর উপর ভিত্তি করে গঠিত, সেটিও reactive। কারণ যখন props
বা state
পরিবর্তন হয়, কম্পোনেন্ট আবার রেন্ডার হয় — তখন সেই ভ্যারিয়েবলগুলোর মানও নতুন করে গণনা হয়।
🧠 উদাহরণ:
function ChatRoom({ roomId, selectedServerUrl }) {
const settings = useContext(SettingsContext); // Context হচ্ছে reactive
const serverUrl = selectedServerUrl ?? settings.defaultServerUrl; // এটি reactive
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, [roomId, serverUrl]); // দুইটা reactive মান এখানে dependency হিসেবে যুক্ত করা হয়েছে
}
ব্যাখ্যা:
roomId
হলো prop → ✅ reactivesettings
হলো context থেকে পাওয়া state → ✅ reactiveserverUrl
হচ্ছে একটি কম্পোনেন্ট বডির ভিতরে তৈরি করা ভ্যারিয়েবল, কিন্তু যেহেতু এটা reactive values দিয়ে তৈরি হয়েছে (selectedServerUrl ও settings), তাই এটি নিজেও ✅ reactive
⚠️ জরুরি নিয়ম:
🔁 React এর Effect রি-রান হয় তখনই, যখন dependency list-এ থাকা কোন reactive মান পরিবর্তন হয়।
🧾 তাই তোমার useEffect
যদি কোন কম্পোনেন্টের ভিতরের মান পড়ে (যেটা রেন্ডারিংয়ের সময় তৈরি হয়), তাহলে সেটা useEffect
এর dependency list-এ রাখতে হবে।
💡 এক কথায় মনে রাখো:
📌 কম্পোনেন্ট বডির ভিতরে যত মান ডিক্লেয়ার করা হয় (props, state, context, বা তাদের ভিত্তিতে তৈরি ভ্যারিয়েবল)—সবকিছুই reactive!
তাই Effect এ যদি তুমি এগুলোর কোনটিকে ব্যবহার করো, তাহলে তা dependency list-এ রাখা বাধ্যতামূলক।
❌ কোন মানগুলো reactive না?
React এর রেন্ডারিং সিস্টেমের বাইরে থেকে আসা যেসব মান আছে, যেমন:
❌ Global Mutable Values:
window.location.pathname; // এটা reactive না
কারণ এটা React এর রেন্ডার সাইকেল অনুযায়ী বদলায় না। তুমি এটা dependency list-এ রাখলেও React বুঝবে না কখন এটা বদলালো।
👉 এই ধরনের মান ব্যবহার করতে চাইলে useSyncExternalStore
ব্যবহার করো।
❌ Ref ভ্যালু (mutable):
ref.current; // এটা reactive না
ref.current
নিজেই একটা mutable object। এটি পরিবর্তিত হলেও component রেন্ডার ট্রিগার হয় না, তাই useEffect
এ dependency হিসেবে রাখলে লাভ নেই।
🔍 শেষ কথা:
✍️ তুমি যখনই useEffect
লিখবে, সব সময় লক্ষ্য রাখো তোমার effect কোন মানগুলো পড়ছে। যদি সেগুলো reactive হয় (অর্থাৎ রেন্ডারের সময় বদলাতে পারে), তাহলে অবশ্যই dependency array
তে যোগ করতে হবে।
অবশ্যই ভাই! আমি Section 3.3 এর এই অংশটা বাংলায় সহজভাবে ব্যাখ্যা করছি যেন আপনি পুরোপুরি বুঝতে পারেন:
🔍 Section 3.3:
React কিভাবে চেক করে আপনি সব reactive value dependency হিসেবে দিয়েছেন কিনা
🧠 সমস্যাটা কী?
এই কোডে একটি ভুল আছে:
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, []); // ❌ এখানে সমস্যা!
এই useEffect
-এ dependency array খালি []
, অথচ ভিতরের কোডে serverUrl
আর roomId
ব্যবহার করা হয়েছে — যেগুলো reactive values (মানে, ভবিষ্যতে বদলাতে পারে)।
⚠️ সমস্যা কেন?
roomId
আসে props থেকে:function ChatRoom({ roomId })
→ মানে এটা reactive।serverUrl
আসেuseState
থেকে:const [serverUrl, setServerUrl] = useState(...)
→ এটাও reactive।
👉 কিন্তু আপনি useEffect
-এ বলছেন না যে এই দুইটার উপর depend করে — তাই React ধরে নিচ্ছে এদের মান কখনো বদলাবে না।
🐞 এর ফলে কী সমস্যা হয়?
ধরুন ইউজার roomId
বা serverUrl
বদলালো।
কিন্তু useEffect
-এ এই পরিবর্তনের কোনো হদিস নেই — তাই connection পুরোনো room/server-এ রয়ে যাবে।
React কোনো নতুন connection করবে না, কোনো disconnect-ও না।
➡️ ফলে ইউজার কিছু বদলালেও, React আগেরটাই চালিয়ে যাবে — এটা একটা bug!
🛠️ সঠিক সমাধান কী?
সব reactive value কে dependency array-এ লিখে দিন:
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, [serverUrl, roomId]); // ✅ সমস্যা সমাধান!
এখন যা হবে:
- যখনই
serverUrl
বাroomId
বদলাবে, React Effect টা আগেরটা বন্ধ করে আবার নতুন করে চালাবে। - তাই নতুন room/server অনুযায়ী connection হবে।
🤖 React কিভাবে এই ভুল ধরতে পারে?
React নিজে না, বরং linter (ESLint + React plugin) আপনাকে warning দেয়:
"You used
roomId
andserverUrl
inside useEffect, but didn’t declare them in dependencies."
👉 আপনি যদি VS Code-এ React lint setup করেন, তাহলে এমন warning পাবেন।
Editor setup গাইড দেখতে এখানে ক্লিক করুন » (opens in a new tab)
🎯 মনে রাখার টিপস:
❓ প্রশ্ন | ✅ উত্তর |
---|---|
কবে dependency array-এ মান দিতে হবে? | যখন useEffect এর ভিতর কোন reactive value (props, state, context, etc.) ব্যবহার করছেন |
কোন মানগুলো reactive নয়? | যেমন setState function বা useRef() এর result – এগুলো stable, তাই dependency না দিলেও হবে |
dependency না দিলে কী হবে? | React জানবে না কবে আবার effect চালানো দরকার — bug হতে পারে |
📌 Extra Tip:
🔸 আপনি যদি dependency array-এ setServerUrl
এর মত stable function দেন — সমস্যা হবে না।
React জানে set
functions কখনো বদলায় না।
✅ সংক্ষেপে মনে রাখুন:
আপনি যদি
useEffect
এর ভিতরে এমন কোনো মান ব্যবহার করেন, যেটা পরিবর্তন হতে পারে, তাহলে সেটা dependency array-এ দেওয়া অবশ্যই দরকার — না হলে React আগের ডেটা নিয়েই পড়ে থাকবে।
ভাই, আপনি চাইলে আমি এর জন্য একটি Visual Cheatsheet বা Bengali Explainer Video Script করে দিতে পারি যেন নতুনরা সহজে শিখতে পারে। দরকার হলে জানিয়ে দিয়েন। 🙌 Here’s a Bangla summary (with code highlights and real-life explanations) for Section 3.4: "What to do when you don't want to re-synchronize" from your React book chapter “Lifecycle of Reactive Effects”:
৩.৪: যখন আপনি চান না Effect বারবার চলুক
আগের উদাহরণে আমরা roomId
ও serverUrl
-কে dependency হিসেবে দিয়ে লিন্টারের error ঠিক করেছিলাম। কিন্তু কিছু ক্ষেত্রে আপনি লিন্টারকে প্রমাণ করতে পারেন যে এই ভ্যালুগুলো reactive না—মানে এগুলো কখনো পরিবর্তন হবে না।
🧠 ১ম কৌশল: Component-এর বাইরে নিয়ে যান
const serverUrl = "https://localhost:1234"; // রেন্ডারের সাথে বদলায় না
const roomId = "general"; // রেন্ডারের সাথে বদলায় না
function ChatRoom() {
useEffect(() => {
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, []); // ✅ dependency নাই, কারণ এগুলো reactive না
}
এখানে serverUrl
আর roomId
component-এর বাইরে, তাই React জানে এগুলো কখনো বদলায় না। Effect শুধু একবারই চলবে।
💡 ২য় কৌশল: Effect-এর ভিতরেই ঘোষণা করুন
function ChatRoom() {
useEffect(() => {
const serverUrl = "https://localhost:1234"; // reactive না
const roomId = "general"; // reactive না
const connection = createConnection(serverUrl, roomId);
connection.connect();
return () => connection.disconnect();
}, []); // ✅ dependency নাই
}
এখানে serverUrl
আর roomId
সরাসরি Effect block-এর ভিতরে। এগুলো rendering-এর সময় তৈরি হচ্ছে না, তাই এগুলো reactive না।
⚠️ ভুল পথে লিন্টার চুপ করাবেন না
অনেকে লিন্টারকে চুপ করাতে নিচের মতো করে:
useEffect(() => {
// ...
// 🔴 খারাপ অভ্যাস!
// eslint-ignore-next-line react-hooks/exhaustive-deps
}, []);
এটা React এর নিয়ম ভাঙা! আপনার কোডে বাগ থাকলে তা ঠিক করতে হবে, না যে শুধু লিন্টার চুপ করালেই সমস্যা মিটে যাবে।
✅ কিভাবে সমস্যা সমাধান করবেন?
- আপনার Effect যদি একাধিক কাজ করে, সেটা আলাদা করুন। Split করা শেখা দরকার
- যদি আপনি state/prop এর সাম্প্রতিক মান পড়তে চান কিন্তু সেটা Effect-এর trigger হিসেবে ব্যবহার করতে না চান, তাহলে Effect-কে দু'ভাগ করুন—reactive অংশ Effect-এ রাখুন, আর non-reactive অংশ অন্য জায়গায় সরিয়ে ফেলুন (যেমন: Effect Event)।
- যখনই কোনো object বা function কে dependency করেন, নিশ্চিত হোন সেটা প্রতি render-এ নতুন না হয়। এখানে বিস্তারিত পড়ুন
তোমার লেখা সামারিটা অনেক গুরুত্বপূর্ণ পয়েন্ট স্পর্শ করেছে। নিচে আমি এটাকে পরিপূর্ণ, সহজ ও পরিষ্কার বাংলায় লিখে দিচ্ছি যেন React এর useEffect
এর লাইফসাইকেল এবং রিএক্টিভ আচরণটা আরও ভালোভাবে বোঝা যায়:
✅ Reactive Effect Lifecycle (রিএক্টিভ এফেক্টের জীবনচক্র) — বাংলায় ব্যাখ্যা
React Component-এর useEffect
এর লাইফসাইকেলকে কখনোই সরাসরি Component-এর লাইফসাইকেলের মতো ধরে নেওয়া যাবে না।
📌 কীভাবে useEffect
কাজ করে?
- প্রথমে JSX রেন্ডার হয় – মানে, যখন Component প্রথমবার মাউন্ট হয় তখন UI রেন্ডার হয়।
- তারপর
useEffect
চলে – UI রেন্ডার হবার পরuseEffect
এর মাধ্যমে কিছু Synchronization (যেমন ডাটা ফেচ, সাবস্ক্রিপশন ইত্যাদি) করা হয়। - যখন Component আপডেট হয় বা নতুন Props আসে, তখনও
useEffect
আবার চলতে পারে — তবে আগের কাজ বন্ধ করে নতুন কাজ শুরু করে। একেই বলে clean-up and re-run।
📌 কেন Clean-Up দরকার?
ধরো তুমি একটা চ্যাট রুমে আছো (roomId 'general'
), হঠাৎ অন্য রুমে গেলে (roomId 'tech'
), তখন আগের সার্ভার কানেকশনটা বন্ধ (clean-up) করে নতুনটা খুলতে হবে।
React এটা automatic ভাবে করে না — তোমাকে useEffect
-এর মধ্যে return
দিয়ে আগের কাজটা বন্ধ করতে হবে।
useEffect(() => {
const connection = createConnection(roomId);
connection.connect();
return () => connection.disconnect(); // clean-up
}, [roomId]); // dependency
📌 Component vs Effect Viewpoint
- Component এর দৃষ্টিকোণ: Component মাউন্ট, আপডেট বা আনমাউন্ট হয়।
- Effect এর দৃষ্টিকোণ: কোন data বা value (যেমন
roomId
,props
,state
) পরিবর্তিত হলে Effect synchronize করে — আগের কাজ বন্ধ করে নতুন করে শুরু করে।
🎯 মূল পয়েন্ট:
👉 যখনই dependency (যেমন roomId
, serverUrl
, props.value
) পরিবর্তন হয়, তখন আগের কাজ clean করে নতুনভাবে synchronize করা হয়।
📌 React-এর Strict Mode এ ডাবল Effect:
React ডেভেলপমেন্ট মোডে useEffect
দুইবার চালায়:
- একবার চালায়, আবার বন্ধ করে।
- তারপর আবার চালায়।
এটা করে শুধু চেক করার জন্য — তুমি clean-up ঠিকঠাক করছো কিনা।
📌 কোন variables dependency হবে?
- Component ফাংশনের ভিতরে যেসব variable (
const
,let
,props
,state
) ডিক্লেয়ার করো — সেগুলো সব reactive, মানে এগুলোর পরিবর্তন effect-কে আবার চালাতে পারে। - তাই, এগুলোকেই dependency হিসেবে
useEffect
এ declare করতে হবে।
useEffect(() => {
doSomething(props.value);
}, [props.value]); // ✅ dependency দিলেই warning যাবে
📌 কখন একাধিক useEffect
লাগবে?
- যদি এক
useEffect
এর কাজ অন্যuseEffect
কে প্রভাবিত করে, তাহলে তাদের আলাদা করা ভালো। - যেমন, তুমি চ্যাট রুম লোড করছো এক
useEffect
এ, আর ইউজারের উপস্থিতি ট্র্যাক করছো আরেকটাতে — তাহলে এগুলো আলাদাuseEffect
হওয়া উচিত।
useEffect(() => {
// Load chat room
}, [roomId]);
useEffect(() => {
// Track user activity
}, [userId]);
🔁 Recap (সারাংশ):
useEffect
UI রেন্ডার হওয়ার পর চালানো হয়।- যেসব ভ্যালু পরিবর্তনশীল (reactive), তাদের dependency হিসেবে declare করতেই হবে।
- আগের effect clean না করলে bug হবে (যেমন multiple API call বা memory leak)।
- effect গুলোকে component-এর দৃষ্টিতে না দেখে, independent synchronization process হিসেবে ভাবো।
- Strict Mode-এ React দুইবার চালিয়ে clean-up টেস্ট করে।
🧍♂️ উদাহরণ: ধরুন আপনি একটা দোকানে আছেন (component)। আপনি কাউকে বলেন, “যখনই দাম বদলাবে বা নতুন পণ্য আসবে, তখন আমাকে জানাবেন” (effect)। এখন যদি আপনি ভুলে যান দাম আর পণ্যের নাম বলতে, তাহলে কেউ আপনাকে আর খবর দেবে না—even যদি সব বদলে যায়। তাই Effect-কে dependency গুলো বলা খুব জরুরি!