كان البحث عن ‘الأقرب’ يجمّد الخريطة: كيف أنقذتنا الأشجار الرباعية (Quadtrees) من جحيم القوة الغاشمة؟

يا جماعة الخير، السلام عليكم ورحمة الله وبركاته. معكم أخوكم أبو عمر.

خلوني أحكيلكم قصة صارت معي قبل كم سنة، قصة فيها قهوة كتير وسهر للصبح، بس نهايتها كانت حلوة وفيها درس كبير. كنا بنشتغل على مشروع تطبيق توصيل لمطعم حلويات مشهور في المدينة، التطبيق فيه خريطة بتعرض كل السائقين المتاحين (الكباتن). صاحب المطعم كان عنده طلب بسيط، بس هالطلب البسيط كان رح يجلطنا: “بدي لما تيجي طلبية جديدة، أكبس كبسة أعرف مين أقرب كابتن عليها عشان أبعته فورًا”.

قلنا له “بسيطة، من عيونا يا حجي”. وبكل بساطة، زميلي في الفريق كتب الكود الأولي. الفكرة كانت واضحة: لما نكبس الكبسة، بنجيب موقع الطلبية الجديدة، وبنعمل حلقة (loop) على كل السائقين المتاحين، نحسب المسافة بين كل سائق والطلبية، ونخزن أقصر مسافة شفناها. منطقي، صح؟

شغلنا التطبيق التجريبي وكان فيه 10 سائقين، الكبسة كانت شغالة زي الصاروخ. فرحنا وقلنا “تمام، هي خلصت”. المشكلة بلشت لما زدنا عدد السائقين في الاختبار لـ 5000 سائق عشان نحاكي وقت الذروة. كبسنا الكبسة… والتطبيق تجمّد. تجمد بمعنى الكلمة، الشاشة بطلت تستجيب، وصار يطلع “Application Not Responding”. صابنا إحباط ما بعده إحباط. يا زلمة، كيف عملية بسيطة زي هاي تعمل كل هالمشكلة؟ قعدنا نفكر ونحلل، وهون بلش جحيم “القوة الغاشمة” يظهر على حقيقته.

جحيم القوة الغاشمة (The Brute-Force Hell)

الطريقة اللي اتبعناها في البداية اسمها في علم الحاسوب “القوة الغاشمة” أو Brute Force. هي أبسط طريقة لحل أي مشكلة: جرّب كل الحلول الممكنة. في حالتنا، “الحلول الممكنة” كانت كل السائقين على الخريطة.

لو عندك 10 سائقين، الكومبيوتر بيعمل 10 عمليات حساب مسافة. ولا إشي. لو عندك 5000 سائق، بيعمل 5000 عملية. لو عندك 50,000؟ بيعمل 50,000 عملية في كل مرة بتكبس فيها الكبسة. هذا بنسميه تعقيد زمني خطي أو O(n)، حيث n هو عدد النقاط. ومع أن الـ 50,000 عملية مش رقم كبير لأي معالج حديث، إلا إن تنفيذها على الواجهة الأمامية (Frontend) مع إعادة رسم الخريطة كان كابوسًا حقيقيًا يسبب تجميد واجهة المستخدم.

الكود الشرير (مثال بسيط)

عشان تتوضح الصورة، هاي نسخة مبسطة من الكود اللي كان عاملنا المشكلة (مكتوب بلغة JavaScript):


function findNearestBruteForce(targetPoint, allPoints) {
  let bestPoint = null;
  let minDistance = Infinity; // ابدأ بأكبر مسافة ممكنة

  for (const currentPoint of allPoints) {
    // حساب المسافة الإقليدية (بدون أخذ الجذر التربيعي للتحسين)
    const distSq = 
      Math.pow(targetPoint.x - currentPoint.x, 2) +
      Math.pow(targetPoint.y - currentPoint.y, 2);

    if (distSq < minDistance) {
      minDistance = distSq;
      bestPoint = currentPoint;
    }
  }

  return bestPoint;
}

// الاستخدام
// const drivers = [ {x: 10, y: 20}, {x: 12, y: 25}, ... 5000 more points ];
// const newOrder = {x: 40, y: 42};
// const nearestDriver = findNearestBruteForce(newOrder, drivers); // This freezes the UI!

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

شجرة الخير: الدخول إلى عالم الأشجار الرباعية (Quadtrees)

بعد ليلة من البحث والقهوة، وتذكر أيام الجامعة ومادة هياكل البيانات، لمعت الفكرة في بالي: “البيانات المكانية إلها حلولها الخاصة!”. وهنا كانت الإجابة: الأشجار الرباعية أو Quadtrees.

شو هي الـ Quadtree؟ باختصار شديد، هي هيكل بيانات شجري مصمم خصيصًا لتقسيم وإدارة البيانات في مساحة ثنائية الأبعاد (مثل خريطتنا). تخيل معي الخريطة كاملة كمربع كبير. الشجرة الرباعية بتشتغل كالتالي:

  • إذا كان هذا المربع يحتوي على عدد قليل من النقاط (أقل من سعة معينة، مثلا 4 نقاط)، خلص، منوقف هنا.
  • أما إذا كان المربع “مزدحم” (فيه نقاط أكثر من السعة المحددة)، فإحنا بنقسمه لأربع مربعات أصغر متساوية (شمال غربي، شمال شرقي، جنوب غربي، جنوب شرقي).
  • بعدين، بنوزع النقاط اللي كانت في المربع الكبير على المربعات الأربعة الجديدة، كل نقطة حسب مكانها.
  • بنكرر هاي العملية لكل مربع جديد بشكل تعاودي (recursive) لحد ما كل المربعات الفرعية تصير “غير مزدحمة”.

النتيجة النهائية بتكون عندك بنية تشبه الشجرة، بتمثل الخريطة مقسمة لمناطق أصغر وأصغر في الأماكن المزدحمة بالنقاط.

كيف أنقذتنا من التجميد؟

الجمال في الـ Quadtree مش في التقسيم نفسه، بل في طريقة البحث الذكية اللي بتوفرها. لما بدك تبحث عن أقرب سائق لنقطة طلبية جديدة، ما في داعي تفحص كل الـ 5000 سائق. اللي بتعمله هو:

  1. بتنزل في الشجرة لحد ما توصل للمربع الصغير (الورقة أو Leaf) اللي بتقع فيه نقطة الطلبية.
  2. بتفترض مبدئيًا إن أقرب نقطة هي أي نقطة موجودة في هذا المربع الصغير. بتحسب المسافة وبتعتبرها “أفضل مسافة حالية”.
  3. وهنا السحر: بترسم دائرة وهمية حوالين نقطة الطلبية، نصف قطرها هو “أفضل مسافة حالية”.
  4. الآن، بدل ما تبحث في كل الخريطة، أنت فقط محتاج تبحث في المربعات الأخرى اللي بتتقاطع مع هاي الدائرة الوهمية.

ليش هذا ذكي؟ لأنه لو في مربع بعيد جدًا عن نقطة الطلبية (أبعد من نصف قطر الدائرة الوهمية)، فمستحيل أصلًا يكون فيه أي نقطة أقرب من النقطة اللي لقيتها بالفعل! وبهذه الطريقة، الخوارزمية “بتتجاهل” آلاف النقاط البعيدة بدون ما تحسب المسافة إلها أصلًا، وهذا هو سر السرعة الخارقة.

التعقيد الزمني للبحث في شجرة رباعية متوازنة هو O(log n) في المتوسط. الفرق بين O(n) و O(log n) هو الفرق بين تجميد التطبيق واستجابة فورية.

مثال كود مبسط للشجرة الرباعية

هذا الكود أعقد شوي، لكنه بيوضح الفكرة الأساسية. (هذا مجرد هيكل توضيحي وليس تطبيقًا كاملاً).


// الهياكل الأساسية
class Point { constructor(x, y, data) { /* ... */ } }
class Rectangle { constructor(x, y, w, h) { /* ... */ } }

class QuadTree {
  constructor(boundary, capacity) {
    this.boundary = boundary; // Rectangle object
    this.capacity = capacity; // أقصى عدد نقاط قبل التقسيم
    this.points = []; // النقاط في هذا المربع
    this.divided = false;
  }

  subdivide() {
    // قسم المربع الحالي إلى أربعة مربعات فرعية
    // nw, ne, sw, se (شمال غربي، شمال شرقي...)
    let { x, y, w, h } = this.boundary;
    let hw = w / 2;
    let hh = h / 2;

    let nw = new Rectangle(x, y, hw, hh);
    this.northwest = new QuadTree(nw, this.capacity);
    // ... and so on for ne, sw, se
    this.divided = true;
  }

  insert(point) {
    // إذا النقطة ليست ضمن حدود هذا المربع، تجاهلها
    if (!this.boundary.contains(point)) {
      return false;
    }

    // إذا كان هناك مكان، أضف النقطة
    if (this.points.length < this.capacity) {
      this.points.push(point);
      return true;
    } else {
      // إذا كان المربع ممتلئًا ولم يتم تقسيمه بعد، قسمه
      if (!this.divided) {
        this.subdivide();
      }
      // ثم حاول إضافة النقطة إلى أحد المربعات الفرعية
      this.northwest.insert(point);
      this.northeast.insert(point);
      this.southwest.insert(point);
      this.southeast.insert(point);
    }
  }

  // دالة البحث (query) بتكون أعقد، وبتبحث بشكل تعاودي
  // فقط في المربعات اللي ممكن تحتوي على نقاط قريبة
  query(range, found) {
    // ... a lot of logic here to prune the search
  }
}

نصيحة من أبو عمر: لا تخاف من تعقيد الكود. الفكرة هي الأهم. اليوم، هناك مكتبات جاهزة تقوم بكل هذا العمل عنك في معظم لغات البرمجة (مثل d3-quadtree في JavaScript). المهم أن تفهم “لماذا” و”متى” تستخدمها.

نصائح عملية من خبرتي

بعد ما استخدمنا الأشجار الرباعية في مشاريع مختلفة، تجمعت عندي شوية نصائح بحب أشاركها معكم:

  • السعة (Capacity) مهمة: اختيار الـ `capacity` (عدد النقاط قبل التقسيم) يؤثر على الأداء. قيمة صغيرة (مثل 1) بتعطي شجرة عميقة جدًا. قيمة كبيرة جدًا بتخلي الشجرة أقرب للقوة الغاشمة. عادةً، قيمة بين 4 و 8 هي بداية جيدة، وتحتاج لتجربة حسب طبيعة بياناتك.
  • البيانات المتحركة: في تطبيق السائقين، مواقعهم بتتغير كل ثانية. تعديل الشجرة (حذف وإضافة نقطة) ممكن، لكنه معقد. الحل الأسهل والأكثر عملية اللي اتبعناه؟ كل 3-5 ثواني، كنا نمسح الشجرة القديمة ونبني شجرة جديدة بالكامل من المواقع المحدثة. بناء الشجرة سريع جدًا، وهذا الحل كان أسهل وأقل عرضة للأخطاء.
  • التخيل هو مفتاح الفهم: لما كنت أتعلمها أول مرة، أفضل إشي ساعدني هو إني أرسمها. استخدمت مكتبة رسم بسيطة على الـ Canvas في HTML ورسمت المربعات والنقاط. لما تشوف بعينك كيف الشجرة بتتقسم وكيف البحث بيتجاهل مناطق كاملة، الفكرة بتترسخ في عقلك للأبد.

الخلاصة والزبدة

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

خوارزمية القوة الغاشمة مغرية ببساطتها، لكنها غالبًا ما تكون فخًا يؤدي إلى مشاكل أداء كارثية في المستقبل. هياكل البيانات المتخصصة مثل الأشجار الرباعية (Quadtrees) قد تبدو مخيفة في البداية، لكنها الأدوات الصحيحة للمشاكل الصحيحة، وهي ما يميز المبرمج الخبير عن المبتدئ.

فيا صديقي المبرمج، في المرة القادمة التي تواجه فيها مشكلة مع مجموعة كبيرة من البيانات، توقف لحظة قبل أن تكتب أول `for` loop يخطر في بالك. اسأل نفسك: “هل هناك هيكل بيانات أذكى لهذه المهمة؟”. الإجابة على هذا السؤال قد توفر عليك ليالي طويلة من السهر والقهوة. 🚀

أبو عمر

سجل دخولك لعمل نقاش تفاعلي

كافة المحادثات خاصة ولا يتم عرضها على الموقع نهائياً

آراء من النقاشات

لا توجد آراء منشورة بعد. كن أول من يشارك رأيه!

آخر المدونات

​معمارية البرمجيات

من الانهيار المتسلسل إلى المرونة: كيف أنقذتنا هندسة الأحداث (EDA) من جحيم التشابك؟

أتذكرها جيداً، ليلة إطلاق الميزة الجديدة في متجرنا الإلكتروني. كان كل تغيير بسيط يهدد بانهيار النظام بأكمله كأحجار الدومينو. في هذه المقالة، أسرد لكم يا...

7 مايو، 2026 قراءة المزيد
ذكاء اصطناعي

كانت أفكارنا سجينة الأوامر المتتالية: كيف حررتنا ‘العملاء المستقلون’ (AI Agents) من جحيم التنفيذ اليدوي؟

مقالة من منظور مبرمج خبير، تستعرض كيف أن العملاء المستقلين (AI Agents) يمثلون نقلة نوعية في عالم البرمجة والأتمتة. من خلال قصة شخصية وأمثلة عملية،...

7 مايو، 2026 قراءة المزيد
تسويق رقمي

كانت قراراتنا التسويقية عمياء: كيف أنقذتنا ‘نمذجة الإحالة متعددة اللمسات’ من جحيم ‘النقرة الأخيرة’؟

كنا على وشك اتخاذ قرارات كارثية بناءً على بيانات مضللة. في هذه المقالة، أسرد لكم قصة كيف أنقذتنا نمذجة الإحالة متعددة اللمسات (Multi-Touch Attribution) من...

7 مايو، 2026 قراءة المزيد
تجربة المستخدم والابداع البصري

كان تصميمنا يستبعد الملايين: كيف أنقذتنا ‘إمكانية الوصول’ (Accessibility) من جحيم التصميم الإقصائي؟

في هذه المقالة، أشارككم قصة حقيقية من مسيرتي كمطور برمجيات، حيث كاد مشروعنا أن يفشل فشلاً ذريعاً بسبب تجاهلنا لمبادئ "إمكانية الوصول". سنغوص في تفاصيل...

7 مايو، 2026 قراءة المزيد
الحوسبة السحابية

كانت فاتورة السحابة تلتهم ميزانيتنا: كيف أنقذتنا ‘الـ FinOps’ من جحيم الإنفاق غير المنضبط؟

أشارككم قصة حقيقية من قلب الميدان، كيف تحولنا من فريق يغرق في فواتير الحوسبة السحابية الباهظة إلى فريق يمتلك زمام المبادرة والتحكم. هذه ليست مجرد...

7 مايو، 2026 قراءة المزيد
التوظيف وبناء الهوية التقنية

كانت مهاراتي سجينة السيرة الذاتية: كيف أنقذني ‘ملف الإثبات’ من جحيم المشاريع التجريبية

بصفتي أبو عمر، مبرمج فلسطيني، أشارككم قصتي مع المشاريع التجريبية التي لا تعود، وكيف أن بناء "ملف إثبات" (Portfolio of Proof) حقيقي كان طوق النجاة...

7 مايو، 2026 قراءة المزيد
التوسع والأداء العالي والأحمال

كانت خوادمنا تغرق تحت الضغط: كيف أنقذتنا ‘موازنة الأحمال’ (Load Balancing) من جحيم النقاط الساخنة؟

في ليلة إطلاق حاسمة، كادت خوادمنا أن تنهار تحت ضغط المستخدمين. أشارككم قصة كيف أنقذتنا تقنية "موازنة الأحمال" من كارثة محققة، وأغوص معكم في تفاصيلها...

7 مايو، 2026 قراءة المزيد
البودكاست