كانت مسارات التوصيل لدينا كابوساً: كيف أنقذتنا خوارزمية A* من جحيم التكاليف الزائدة؟

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

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

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

في هداك اليوم، ابتسمت وقلتله: “طوّل بالك يا أبو أحمد، الحكي بينا، شكلها المشكلة مش في السواقين، المشكلة بدها شوية كود… بدها خوارزمية تفهم شغلنا”. ومن هنا بدأت رحلتنا مع المنقذ: خوارزمية A*.

لماذا كانت الطرق التقليدية كابوساً؟

قبل ما نحكي عن الحل، خلينا نفصّل المشكلة أكثر. المشكلة لم تكن مجرد “إيجاد طريق” من النقطة أ إلى النقطة ب. المشكلة كانت في إيجاد الطريق الأمثل. الاعتماد على التخطيط اليدوي أو حتى على أدوات الخرائط لكل رحلة على حدة كان فاشلاً للأسباب التالية:

  • انعدام النظرة الشاملة: كان كل سائق يرى فقط مهمته التالية، دون وجود نظام ينسق بين جميع السائقين لتحقيق الكفاءة القصوى للأسطول بأكمله.
  • التكلفة الحسابية البشرية: تخيل أن لديك 50 طلبًا و 5 سائقين. عدد الطرق الممكنة لترتيب هذه الزيارات فلكي! الدماغ البشري لا يمكنه معالجة كل هذه الاحتمالات للعثور على الأفضل.
  • عوامل متغيرة: المسافة ليست كل شيء. ماذا عن الوقت؟ الازدحام المروري؟ أولويات التسليم؟ الطرق التقليدية كانت تتجاهل هذه العوامل المعقدة.

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

المنقذ: خوارزمية A*… شو قصتها؟

خوارزمية A* (تلفظ “A-Star” أو “إيه ستار”) هي واحدة من أشهر وأقوى خوارزميات البحث عن المسار (Pathfinding). هي خوارزمية ذكية تجمع بين دقة خوارزمية “دايكسترا” (Dijkstra’s Algorithm) وسرعة الخوارزميات الجشعة (Greedy Algorithms). فكر فيها كأنها سائق محترف جدًا: هو لا يعرف فقط الطرق التي سلكها، بل لديه أيضًا “حدس” قوي حول أفضل طريق للوصول إلى وجهته النهائية.

المعادلة السحرية: f(n) = g(n) + h(n)

سر عبقرية A* يكمن في هذه المعادلة البسيطة. خلينا نفصصها على بلاطة:

  • g(n): هذه هي التكلفة الفعلية للوصول من نقطة البداية إلى النقطة الحالية (n). بالبلدي، هاي “الفاتورة اللي دفعناها لهلأ”. ممكن تكون المسافة المقطوعة بالكيلومترات أو الوقت المستغرق بالدقائق.
  • h(n): هذه هي الدالة الحدسية (Heuristic Function). هي تقدير وتخمين للتكلفة من النقطة الحالية (n) إلى نقطة النهاية. هذا هو “الحدس” أو “النظرة المستقبلية” للخوارزمية. إنها تخمين ذكي، وليست القيمة الحقيقية.
  • f(n): هي المجموع الكلي التقديري للمسار. الخوارزمية في كل خطوة تختار النقطة التي لديها أقل قيمة `f(n)`، أي النقطة التي تبدو واعدة أكثر.

هذا المزيج بين التكلفة الفعلية `g(n)` والتكلفة المستقبلية المخمنة `h(n)` هو ما يجعل A* فعالة جدًا. هي لا تندفع بشكل أعمى نحو الهدف مثل الخوارزميات الجشعة، ولا تستكشف كل اتجاه ممكن ببطء مثل دايكسترا.

من النظرية إلى التطبيق: كيف بنينا الحل خطوة بخطوة

الحكي النظري حلو، بس خلينا نشوف كيف طبقنا هذا الحكي على أرض الواقع. “يلا نبرمج!”.

1. تمثيل الخريطة كمخطط (Graph)

أول خطوة كانت تحويل خريطة المدينة إلى شيء يفهمه الكمبيوتر. قمنا بتمثيل كل موقع مهم (المستودع، منزل كل عميل) كـ “عقدة” (Node) في مخطط بياني. والطرق التي تربط بين هذه المواقع أصبحت “حواف” (Edges). كل حافة أعطيناها “وزن” (Weight) يمثل التكلفة، والذي كان في حالتنا هو المسافة بالكيلومترات.

2. اختيار دالة التكلفة (g(n))

هذه كانت سهلة. `g(n)` هي ببساطة مجموع أوزان الحواف (المسافات) التي قطعناها للوصول إلى العقدة الحالية `n`.

3. فن اختيار الدالة الحدسية (h(n))

هنا يكمن التحدي والذكاء. دالة `h(n)` يجب أن تكون “متفائلة”، أي أنها يجب ألا تبالغ أبدًا في تقدير التكلفة المتبقية. إذا بالغت في التقدير، قد تفوت الخوارزمية المسار الأمثل. أفضل وأبسط دالة حدسية في حالتنا كانت:

المسافة الإقليدية (Euclidean Distance): وهي المسافة المستقيمة “كما يطير الغراب” بين النقطة الحالية والهدف. هذه المسافة هي دائمًا أقصر من أو تساوي المسافة الفعلية على الطرق، لذا فهي تخمين متفائل ومثالي لـ A*.

4. مثال كود بسيط (Python)

لتقريب الصورة، إليك مثال مبسط جدًا بلغة بايثون يوضح فكرة عمل A*. سنحتاج إلى مكتبة `heapq` لعمل طابور الأولوية (Priority Queue) الذي يجعل الخوارزمية سريعة.


import heapq

def a_star(graph, start_node, goal_node):
    # قائمة العقد التي لم تتم زيارتها بعد (طابور أولوية)
    # كل عنصر هو (f_cost, g_cost, node, path)
    open_list = [(0, 0, start_node, [start_node])]
    
    # مجموعة لتخزين العقد التي تمت زيارتها لعدم تكرارها
    closed_set = set()

    while open_list:
        # احصل على العقدة ذات أقل f_cost
        f_cost, g_cost, current_node, path = heapq.heappop(open_list)

        # إذا وصلنا للهدف، أرجع المسار والتكلفة
        if current_node == goal_node:
            return path, g_cost

        # إذا زرنا هذه العقدة من قبل بمسار أفضل، تجاهلها
        if current_node in closed_set:
            continue
        
        # أضف العقدة الحالية إلى قائمة الزيارة
        closed_set.add(current_node)

        # استكشف الجيران
        for neighbor, distance in graph[current_node].items():
            if neighbor in closed_set:
                continue

            # حساب التكاليف الجديدة
            new_g_cost = g_cost + distance
            
            # الدالة الحدسية (هنا سنفترض وجود دالة heuristic(node, goal))
            h_cost = heuristic(neighbor, goal_node) 
            new_f_cost = new_g_cost + h_cost

            # أضف الجار إلى القائمة المفتوحة
            heapq.heappush(open_list, (new_f_cost, new_g_cost, neighbor, path + [neighbor]))

    # إذا لم يتم العثور على مسار
    return None, float('inf')

# مثال بسيط للدالة الحدسية (المسافة المستقيمة)
# في الواقع، ستحسبها بناءً على إحداثيات (خطوط الطول والعرض)
def heuristic(node, goal):
    # هذا مجرد مثال، يجب أن يكون الحساب الفعلي بناءً على بيانات حقيقية
    heuristics = {
        'A': 10, 'B': 5, 'C': 3, 'D': 0 # المسافة التقديرية من كل عقدة للهدف D
    }
    return heuristics.get(node, float('inf'))

# مثال على المخطط (الخريطة)
# graph = { 'A': {'B': 5, 'C': 2}, ... }
# path, cost = a_star(graph, 'A', 'D')

النتائج على أرض الواقع: من الفوضى إلى النظام

بعد تطبيق النظام الجديد، كانت النتائج مذهلة وفاقت التوقعات:

  • انخفاض تكاليف الوقود: لاحظنا انخفاضًا بنسبة تقارب 25% في استهلاك الوقود خلال الشهر الأول فقط، لأن المسارات أصبحت أقصر وأكثر منطقية.
  • زيادة سرعة التوصيل: متوسط وقت التوصيل قل بشكل ملحوظ. الزبائن صاروا يستلموا طلباتهم بشكل أسرع.
  • عملاء أكثر سعادة: الشكاوى من التأخير شبه اختفت. بل وبدأنا نحصل على تقييمات إيجابية تشيد بسرعة ودقة التوصيل.
  • راحة نفسية لفريق العمل: أبو أحمد نفسه صار يجي الصبح يكبس زر واحد، والنظام يولد أفضل المسارات لكل سائق. صار وقته متاح لمتابعة أمور استراتيجية أهم.

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

إذا كنت تفكر في استخدام A* أو خوارزميات مشابهة، اسمح لي أن أقدم لك بعض النصائح من “كيس” الخبرة:

  1. الدالة الحدسية هي الملك: جودة الدالة الحدسية `h(n)` هي التي تحدد سرعة وفعالية الخوارزمية. استثمر وقتًا في تصميم دالة حدسية جيدة ومتفائلة (admissible).
  2. A* ليست الحل لكل شيء: خوارزمية A* رائعة لإيجاد أقصر طريق بين نقطتين (أ و ب). لكن في مشكلة التوصيل الحقيقية، أنت تتعامل مع “مشكلة البائع المتجول” (Traveling Salesman Problem – TSP)، حيث تحتاج لزيارة عدة نقاط بأقصر مسار إجمالي. في هذه الحالة، A* تكون أداة مساعدة ضمن حل أكبر. يمكنك استخدام A* لحساب المسافات بين كل نقطتين، ثم استخدام خوارزمية أخرى (مثل خوارزمية الجار الأقرب أو خوارزميات جينية) لترتيب زيارة النقاط.
  3. البيانات هي الواقع: جودة حلك تعتمد على جودة بياناتك. تأكد من أن خريطتك (المخطط البياني) تعكس الواقع قدر الإمكان. هل هناك طرق باتجاه واحد؟ هل هناك قيود على أوزان الشاحنات في شوارع معينة؟ كلما كانت بياناتك أدق، كانت نتائجك أفضل.

الخلاصة: الخوارزمية ليست مجرد كود، بل هي طريقة تفكير

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

فلا تستهينوا بقوة الخوارزميات. ابحثوا عن “وجعة الراس” في عملكم أو في محيطكم، وفكروا… هل يمكن لخوارزمية ذكية أن تكون هي الحل؟ صدقوني، في كثير من الأحيان، الجواب هو نعم. 💪

يلا شدوا حيلكم، وخلي الكود يحكي عنكم!

أبو عمر

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

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

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

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

آخر المدونات

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

من جحيم التبعيات إلى نعيم الاستقلالية: رحلتي مع المعمارية القائمة على الأحداث (EDA)

كانت خدماتنا متشابكة في تبعيات قاتلة تحوّل كل تحديث إلى كابوس. في هذه المقالة، أروي لكم كيف حررتنا "المعمارية القائمة على الأحداث" (Event-Driven Architecture) من...

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

كانت مراجعات المستخدمين صراخاً في الظلام: كيف أنقذنا ‘تحليل المشاعر’ من جحيم تجاهل صوت العميل؟

في عالم تتكدس فيه آراء المستخدمين بالآلاف، يصبح تجاهلها جحيماً حقيقياً. أسرد لكم قصتي كـ "أبو عمر"، وكيف تحولنا من ضياع تام في بحر المراجعات...

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

ميزانية التسويق كانت ثقباً أسود: كيف أنقذتنا ‘نماذج الإحالة المبنية على البيانات’ من جحيم التخمين؟

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

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

كانت واجهاتنا متحفاً للفوضى: كيف أنقذنا ‘نظام التصميم’ (Design System) من جحيم التناقض البصري؟

أشارككم قصة حقيقية من قلب المعركة البرمجية، يوم كانت واجهاتنا عبارة عن فوضى بصرية عارمة. اكتشفوا معنا كيف استطعنا، بفضل "نظام التصميم"، تحويل هذا الجحيم...

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

كانت إجاباتي ‘عن نفسي’ كارثية: كيف أنقذتني طريقة STAR من جحيم المقابلات السلوكية؟

أشارككم تجربتي الشخصية مع المقابلات السلوكية وكيف تحولت من الفشل الذريع إلى النجاح بفضل طريقة STAR. هذه المقالة دليلكم العملي لتجاوز أسئلة "أخبرني عن موقف..."...

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

طلبات المستخدمين كانت تضيع: كيف أنقذتنا “طوابير الرسائل” (Message Queues) من جحيم فقدان البيانات؟

أشارككم قصة حقيقية من قلب المعركة، عندما كادت طلبات المستخدمين أن تضيع في زحام الضغط الهائل. اكتشفوا كيف أنقذنا مفهوم "طوابير الرسائل" (Message Queues) من...

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