بايثونOverriding
- مفهوم إعادة التعريف
- مثال حول إعادة التعريف
- الدالة super()
مفهوم إعادة التعريف
في الدرس السابق، تعلمنا كيفية جعل الكلاس يرث المتغيرات و الدوال الموجودة في كلاس آخر بالإضافة إلى إخفاء البيانات في الكلاس الأب حتى لا يحدث تضارب في الأسماء.
إعادة التعريف أو التعريف من جديد ( Overriding ) تعني أن يقوم الكلاس الإبن بإعادة تعريف نفس الدالة التي ورثها من الكلاس الأب. الدالة الجديدة تكون مشابهة للدالة الموروثة من حيث الشكل فقط، أي تملك نفس الإسم و عدد الباراميترات لكن محتواها مختلف.
الهدف الحقيقي من إعادة التعريف هو إتاحة الفرصة للكلاس الإبن ليعرّف الدوال حسب حاجته.
في دروس لاحقة، سنرث من كلاسات جاهزة في بايثون و نقوم بإعادة تعريف الدوال التي نرثها لكي تناسب التطبيقات التي نبنيها.
شروط إعادة التعريف
- لا يمكن إعادة تعريف دالة معرّفة بالأساس بشكل خفي.
- عدد و نوع باراميترات الدالة الجديدة يجب أن يطابق عدد و نوع باراميترات الدالة القديمة.
عدم تحقق هذين الشرطين يجعل مفسر بايثون يفهم أننا نريد إضافة دالة جديدة في الكلاس الإبن و ليس إعادة تعريف الدالة التي ورثها في الأصل من الكلاس الأب.
مثال حول إعادة التعريف
في المثال التالي قمنا بتعريف كلاس إسمه CountryInfo و يحتوي على دالة إسمها print_language().
بعدها قمنا بتعريف ثلاث كلاسات ترث من الكلاس CountryInfo مما يعني أن كل واحد منهم يحتوي على الدالة print_language().
هنا الفكرة أن أي كلاس يرث من كلاس CountryInfo قد يضطر إلى تعريف الدالة print_language() من جديد حتى تناسبه.
مثال
نتيجة تشغيل الملف Test
.
Arabic
Spanish
من خلال المثال السابق وضحنا فائدة إعادة تعريف الدالة التي يرثها الكلاس الإبن.
إمكانية الوصول للدالة الأصلية
إذا عدت للكلاس Lebanon،, ستجد أننا أعدنا تعريف الدالة print_language() حتى تطبع Arabic
بدلاً من English
في حال إنشاء كائن من الكلاس Lebanon و إستدعاءها منه.
فعلياً، الكلاس Lebanon أصبح يملك دالتين إٍسمهما print_language() و ليس دالة واحدة كالتالي:
- الأولى هي التي ورثها في الأساس من الكلاس CountryInfo.
- الثانية هي التي قمنا بتعريفها فيه من جديد في الكلاس Lebanon.
بما أن الكلاس Lebanon يملك دالتين لهما نفس الإسم و عدد الباراميترات، كيف عرف مفسّر بايثون أي دالة منهما سينفذ عند تشغيل الكود؟
مفسّر بايثون يبحث في المرتبة الأولى عن الدالة التي قمت باستدعائها في الكلاس الذي تم منه إنشاء الكائن.
إذا وجد الدالة في الكلاس فإنه يقوم بتنفيذها. إن لم يجدها فيه فإنه يبحث عنها في الكلاس الأب له. إن وجدها في الكلاس الأب يقوم بتنفيذها.
في الكلاس Lebanon لا يزال بإمكانك الوصول للدالة print_language() التي ورثها من الكلاس CountryInfo بالإعتماد على الدالة super().
الدالة super()
هذه الدالة تجعلك قادر على الوصول إلى الدالة المعرّفة في الكلاس الأب ( Superclass ) من الكلاس الإبن ( Subclass ) سواء كان يوجد تضارب في الأسماء أو لا يوجد و بالتالي يصبح الكلاس الإبن قادر على أن يعيد كتابة الدالة التي ورثها من الكلاس الأب و أن يظل قادر على الوصول إلى الدالة التي ورثها من الكلاس الأب أيضاً.
بمعنى آخر فإن هذه الدالة أن تجعل المبرمج قادر على الإستفادة من الدالة الموجودة في الكلاس الأب و إضافة أشياء جديدة عليها في الكلاس الإبن.
ستتعلم من المثال التالي كيف تقوم باستدعاء الدالة __init__() الموجودة في الكلاس الأب بشكل تلقائي من الدالة __init__() الموجودة في الكلاس الإبن.
تذكر أن الباراميترات التي يتم تعريفها بداخل الدالة __init__() يتم تحويلها إلى خصائص للكائن الذي يتم إنشاؤه و هذه الدالة تستدعى بشكل تلقائي عند إنشاء الكائن.
في المثال التالي قمنا بتعريف كلاس إسمه Person يحتوي على خاصيتين هما name و age.
هاتين الخاصيتين يتم إنشاؤهما في لحظة إنشاء كائن من الكلاس لأننا وضعناهما في الدالة __init__().
بعدها قمنا بتعريف كلاس إسمه Student يرث من الكلاس Person.
في هذا الكلاس قمنا بإعادة تعريف الدالة __init__() و وضعنا فيها نفس الباراميترات الموجودة في الدالة __init__() التي ورثها الكلاس و أضفنا فيها باراميتر جديد إسمه specialization. بعدها قمنا بتعريف دالة أخرى إسمها print_info() تعرض كل قيم الخصائص الموجودة في الكلاس.
مثال
نتيجة تشغيل الملف Test
.
age: 24
specialization: Computer Science