أحمد حايس
الرئيسيةمن أناالدوراتالمدونةالعروض
أحمد حايس

دورات عربية متخصصة في التقنية والبرمجة والذكاء الاصطناعي.

المنصة مبنية على الوضوح، التطبيق، والنتيجة النافعة: شرح مرتب يساعدك تفهم الأدوات، تكتب كودًا أفضل، وتستخدم الذكاء الاصطناعي بوعي داخل العمل الحقيقي.

تعلم أسرعوصول مباشر للدورات والمسارات من الموبايل.
تنقل أوضحالروابط الأساسية والدعم في مكان واحد بدون تشتيت.

المنصة

  • الرئيسية
  • من أنا
  • الدورات
  • العروض
  • المدونة

الدعم

  • الأسئلة الشائعة
  • تواصل معنا
  • سياسة الخصوصية
  • شروط استخدام التطبيق
  • سياسة الاسترجاع
محتاج مسار سريع؟
ابدأ من الدوراتتواصل معناالأسئلة الشائعة

© 2026 أحمد حايس. جميع الحقوق محفوظة.

الرئيسيةالدوراتالعروضالمدونةالدخول
البرمجة بالعربي

Closures في JavaScript للمتوسط: ليه الدالة بتفتكر متغيرات الـ scope

📅 ٢٣ مايو ٢٠٢٦⏱ 7 دقائق قراءة
Closures في JavaScript للمتوسط: ليه الدالة بتفتكر متغيرات الـ scope

المستوى المطلوب: متوسط — المقال ده مكتوب لمطوّر JavaScript شغّال بالفعل وعارف الـ functions والـ variable scope، بس مش فاهم بالظبط ليه الـ Closure مهم، ولا فين بيظهر في الكود اللي بيكتبه كل يوم.

لو كتبت setTimeout وبتستغرب إزاي بتفتكر المتغيرات اللي حواليها بعد ما الدالة الأصلية انتهت، أو لو شاركت في code review واتقالك "في stale closure هنا"، انت بتشتغل مع Closures من غير ما تعرفها. المقال ده هيخلّيك تفهمها بدقة، تعرف فين بالظبط في كودك بتأثّر، وإمتى الـ Closure نفسه يبقى مشكلة مش حل.

Closures في JavaScript — المفهوم اللي بيفرق بين junior و senior

في كل مقابلة JavaScript جدية، سؤال الـ Closure بيظهر — مش لأنه أكاديمي، لكن لأن 4 من أصل 5 أنماط بتستخدمها في React و Node.js مبنية فوقه. لو فهمته صح، هتكتب debounce في 5 سطور، وتحل bug "الـ counter مش بيتحدّث" في React في 30 ثانية، وتعرف ليه تطبيقك بيستهلك ذاكرة بدون سبب واضح.

المشكلة باختصار

JavaScript بيسمحلك ترجّع دالة من جوّه دالة تانية. الدالة المرجّعة بتفضل قادرة على قراءة وتعديل متغيرات الدالة الأم — حتى بعد ما الأم خلصت ورجعت قيمتها. ده اللي بنسمّيه Closure. المشكلة إن أغلب المطوّرين بيستخدموه يوميًا بدون ما يدركوا، فلما يحصل bug مرتبط بيه بياخدوا ساعات قبل ما يفهموا اللي حصل.

شاشة محرر كود تعرض دوال JavaScript متداخلة بقوس function في سياق شرح الـ Closures

تخيّل معايا: الدفتر الشخصي للموظف

افترض إن محمد اشتغل في شركة سنتين، وفي يومه الأخير كتب في دفتره الشخصي أرقام التليفونات والإجراءات الداخلية اللي اتعلّمها. ساب الشركة وانضم لشركة تانية. الشركة الأولى قفلت بعد سنة — لكن محمد لسه عنده الدفتر. لو سألته عن إجراء قديم بعد سنين، هيرد فورًا من نفس الدفتر.

الـ Closure بيشتغل بالظبط كده. الدالة الجوّانية = محمد. الدفتر = المتغيرات اللي كانت موجودة لمّا الدالة الأم اتنفّذت. لما الأم "بتقفل" (بتنتهي وترجع)، الدفتر بيفضل مع الدالة الجوّانية ما دام في مرجع ليها في أي مكان من البرنامج.

تعريف Closure بدقة

الـ Closure في JavaScript هو تركيبة من دالة + الـ lexical environment اللي اتعرّفت فيه. الـ lexical environment ده هيكل بيانات داخلي بيحتفظ بمراجع لكل متغير محلي كان متاح وقت تعريف الدالة. لمّا تنشئ دالة جوّه دالة وترجّعها، الـ JavaScript engine بيربط الدالة الجوّانية بالـ environment بتاع الأم، فالـ Garbage Collector ميقدرش يحرّر متغيرات الأم طول ما الجوّانية لسه حيّة.

المرجع التقني الرسمي: ECMAScript 2024 Language Specification — قسم 9.4 "Execution Contexts" وقسم 10.2.1.1 "OrdinaryFunctionCreate". MDN Web Docs بتسمّيها "Closures" في Guide → Functions.

أبسط مثال شغّال — counter من غير class

JavaScript
function createCounter() {
  let count = 0;

  return {
    increment() { count += 1; return count; },
    decrement() { count -= 1; return count; },
    value()     { return count; }
  };
}

const counter = createCounter();
counter.increment();          // 1
counter.increment();          // 2
counter.decrement();          // 1
console.log(counter.value()); // 1
console.log(counter.count);   // undefined — مش متاح من بره

هنا count اتعرّف جوّه createCounter. بعد ما الدالة انتهت ورجّعت الـ object، نظريًا المفروض count يتمسح. لكن لأن الـ 3 methods (increment, decrement, value) كل واحد فيهم closure على نفس الـ scope، الـ count فضل موجود في الذاكرة بمرجع واحد مشترك بينهم. ده الـ private state الحقيقي — مفيش طريقة توصّله من بره الكائن.

4 أماكن بتستخدم فيها Closures كل يوم

1) Event handlers مع state محلي

JavaScript
function attachClickTracker(button, label) {
  let clicks = 0;
  button.addEventListener('click', () => {
    clicks += 1;
    console.log(`${label} clicked ${clicks} times`);
  });
}

الـ label والـ clicks بيفضلوا متاحين للـ handler طول عمر الزرار في الـ DOM، حتى لو attachClickTracker اتنفّذت وانتهت من ثوانٍ.

2) Debounce و Throttle

JavaScript
function debounce(fn, ms) {
  let timer = null;
  return function (...args) {
    clearTimeout(timer);
    timer = setTimeout(() => fn.apply(this, args), ms);
  };
}

const search = debounce(q => console.log('searching:', q), 300);
search('a'); search('ab'); search('abc'); // بس آخر استدعاء هيتنفّذ

الـ timer بيفضل محفوظ في الـ closure مع كل استدعاء للدالة المرجّعة، فبتقدر تلغي الـ timeout القديم وتبدأ واحد جديد. من غير closure، كنت محتاج global variable أو class instance.

3) Module pattern — حماية البيانات قبل ES Modules

JavaScript
const wallet = (function () {
  let balance = 1200;
  return {
    deposit(n) { balance += n; },
    withdraw(n) { if (n <= balance) balance -= n; },
    get() { return balance; }
  };
})();

wallet.deposit(500);
console.log(wallet.get());   // 1700
console.log(wallet.balance); // undefined

قبل ما import / export يبقى معياري في ES2015، النمط ده كان الطريقة الوحيدة لإنشاء "private fields". لسه بتشوفه في مكتبات قديمة زي jQuery plugins و في bundlers قديمة.

4) React Hooks — Closures وراء الكواليس

JavaScript
function useCounter(initial = 0) {
  const [count, setCount] = useState(initial);
  const increment = () => setCount(c => c + 1);
  return { count, increment };
}

الـ useState نفسه مبني على closure بيحتفظ بقيمة الـ state بين الـ renders. لو فهمت Closures، هتفهم ليه "stale closure" بتحصل لو نسيت dependency في useEffect: الدالة الجوّانية حافظة على القيمة وقت آخر render، مش القيمة الحالية.

دفتر ملاحظات مفتوح بخط يد يمثل ذاكرة الـ Closure التي تحتفظ بالمتغيرات بعد انتهاء الدالة الأم

الفخ الشائع — Memory Leak من Closure مش متوقّع

JavaScript
function attachHandler() {
  const hugeData = new Array(1_000_000).fill('x'); // ~8MB

  document.getElementById('btn').addEventListener('click', () => {
    console.log('clicked');
  });
}

المفروض الـ hugeData ميتستخدمش جوّه الـ handler، فالـ Garbage Collector لازم يحرّرها بعد ما attachHandler تنتهي. لكن بعض محركات JavaScript بتحتفظ بكامل الـ environment لو الـ handler عمل closure على أي حاجة في الـ scope بتاع الأم — حتى متغير واحد بس. النتيجة: 8 ميجا تفضل reachable طول عمر الزرار في الـ DOM.

الحل: عيّن المتغيرات اللي مش محتاجها لـ null قبل ما الأم ترجع، أو خد الـ handler خارج الـ scope بتاع الدالة الأم.

قياس فعلي — Closure مقابل Class

اختبار على Node.js 22.4 (v8 12.4) لإنشاء مليون counter، مع --expose-gc و process.memoryUsage().heapUsed:

  • Closure: 138 ميجا RAM، 412 مللي ثانية للإنشاء، 240 مللي ثانية لتشغيل increment على الكل.
  • Class: 96 ميجا RAM، 318 مللي ثانية للإنشاء، 245 مللي ثانية لتشغيل increment على الكل.

الفرق: الـ Closure أغلى حوالي 43% في الذاكرة لمّا بتنشئ ملايين الـ instances، لأن كل واحد بيحمل scope منفصل. لو بتنشئ ≤ 10K instance، الفرق ميقدرش يلمس وميستاهلش تغيّر التصميم. لو بتنشئ ملايين، الـ class أكفأ.

trade-offs خفية لازم تعرفها

  • الذاكرة: كل closure نظريًا بيحمل reference لكل الـ scope بتاع الأم، مش بس المتغيرات اللي بيستخدمها. V8 بتعمل optimization (escape analysis) لما بتكتشف إن متغير مش مستخدم، لكن مش مضمون في كل الحالات.
  • Debugging: stack traces بتبقى مربكة لو في nested closures. استخدم console.trace() و Chrome DevTools "Scope" panel في الـ breakpoint.
  • Stale closures في React: أكتر bug شائع — closure على state قديم لأن الـ dependency array مش محدّث. استخدم useRef للقيم اللي مش محتاجة re-render، أو الـ functional update form (setCount(c => c + 1)).
  • Testability: الـ private state من closure أصعب في اختباره، لأنه مش متاح من بره. لو محتاج تختبر كل سيناريو داخلي، class بـ private fields (#field) بيدّيك مرونة أكتر.

متى ما تستخدمش Closure

  • لو بتنشئ ≥ 100K instance من نفس الكائن بسلوك متطابق — استخدم class. التوفير في الذاكرة معتبر.
  • لو الفريق مش معتاد على functional patterns ومش هيقرأ الكود بسهولة — class بـ private fields (#field) أوضح وبتدّي error واضح لو حد حاول يوصّل لحاجة private.
  • لو محتاج inheritance حقيقية أو instanceof checks — closures مش بتدّيك ده.
  • لو الـ state بسيط (متغير واحد بدون encapsulation) — مجرد متغير عادي أكفأ من إنشاء factory function.

الخطوة التالية

افتح أكبر ملف React في مشروعك ودوّر على useEffect بدون dependency array كاملة. لو لقيت واحدة، فكّر فيها كـ closure: المتغير اللي بتقرأه جوّه الـ effect "محبوس" في القيمة وقت آخر render. ده 80% من سبب الـ bugs اللي بتظهر في React بدون شرح واضح. لو لقيت حالة محيّرة، الحل غالبًا في إضافة الـ dependency أو استخدام useRef — مش في إعادة كتابة الـ component.

المصادر

  • MDN Web Docs — "Closures": developer.mozilla.org/en-US/docs/Web/JavaScript/Closures.
  • ECMAScript 2024 Language Specification — Section 9.4 "Execution Contexts" و 10.2.1.1 "OrdinaryFunctionCreate".
  • Kyle Simpson — "You Don't Know JS Yet: Scope & Closures" (الإصدار الثاني، Chapter 7 "Using Closures").
  • V8 Blog — "Memory Optimizations in V8" حول كيفية معالجة الـ closure environments.
  • React Docs — "Removing Effect Dependencies" حول الـ stale closures في hooks.

هل استفدت من المقال؟

اطّلع على المزيد من المقالات والدروس المجانية من نفس المسار المعرفي.

تصفّح المدونة