البرمجة الموجهة للكائنات OOP في Python - جزء1 Class and Object

البرمجة الموجهة للكائنات OOP في Python
جزء1 - Class and Object

 يبرز اسلوب البرمجة الموجهة للكائنات (Object-Oriented Programming - OOP) كأحد الأساليب الرئيسية التي تساعد على تنظيم الكود بشكل فعال .و لغة البرمجة Python واحدة من أهم اللغات التي تدعم هذا الإسلوب... فكيف يتم العمل بالبرمجة الموجهة للكائنات (OOP) مع Python..!!!


في هذا المقال، سنستكشف عالم البرمجة الموجهة للكائنات (OOP) في Python، وسنلقي نظرة عميقة على كيفية استخدام الفئات والكائنات Class & Object. لبناء برامج مرنة ومنظمة. سنتعرف على المفاهيم الأساسية للـ OOP وهي الـ Class و الـ Object وكيفية تطبيقها بشكل عملي في Python...لنبدأ…



عن البرمجة الموجهة للكائنات (OPP) في بايثون

بايثون هي لغة متعددة النماذج، فهي تسمح لنا بالبرمجة بأسلوب إجرائي (Procedural Programming ) وأسلوب البرمجة الموجهة للكائنات (Object-Oriented Programming)  وإسلوب البرمجة الوظيفية (Functional Programming)، كما يمكن البرمجة بمزيج من الأساليب التي تدعمها Python فالبرمجة بلغة بايثون لا تجبرنا على طريقة معينة.

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

OOP هو اختصار يرمز إلى نموذج البرمجة الشيئية أو البرمجة الموجهة للكائنات (Object-Oriented Programming). يتم تعريفه على أنه نموذج برمجة يستخدم مفهوم الكائنات الذي يشير إلى كيانات العالم الحقيقي ذات الحالة والسلوك. للمزيد حول OOP يمكن الإطلاع على مقال عن البرمجة الموجهة للكائنات OOP.

Python من لغات البرمجة التي تدعم نهج البرمجة الموجهة للكائنات OOP. وهذا يجعل من السهل إنشاء واستخدام الفئات (Classes) والكائنات (Objects). ويعتمد العمل مع البرمجة الموجهة بالكائنات OOP على إدراك وعدة مفاهيم برمجية من أهمها التالي:

  • الفئات والكائنات Class & Objects.
  • التغليف Encapsulation.
  • الوراثة Inheritance.
  • تعدد الأشكال Polymorphism.

في هذا المقال و من خلال تعليمات لغة بايثون سنتعرف بالتفصيل على طريقة العمل وتطبيق مفهومي الفئات (Classes ) و الكائنات (Objects) .



الفئات والكائنات Class & Objects

الـ Class هو نموذج يحتوي سمات كائن معين من العالم الحقيقي و سلوك هذه الكائن ، هذه الكائنات ليست مجرد بيانات وليست وظائف فقط، ولكنها مزيج من الاثنين معًا. فكل كائن في العالم الحقيقي له سمات وسلوك مرتبط به. مثلاً :

  • في Class الخاص بالطلاب : تعرف السمات Attributes اسم الطالب وفصله ,والمواد المسجل فيها وعلاماته وما إلى ذلك، بينما يمكن حساب النسبة المئوية لدرجات الطالب بواسطة Method.
  • في Class الخاص بالموظفين : تعرف السمات Attributes باسم الموظف ووظيفته وقسمه وراتبه وما إلى ذلك، بينما حساب الحوافز المستحقة للموظف بواسطة Method.
  • في Class الخاص بالفواتير : رقم الفاتورة، العميل، رمز المنتج واسمه، السعر والكمية، وما إلى ذلك، في الفاتورة.بينما يمكن تطبيق ضريبة السلع والخدمات على قيمة الفاتورة بواسطة Method.

اما الـ Object هو التمثيل الحقيقي للـ Class فمثلاً الـ Class الخاص بالطلاب يمثل الـ Object بيانات لطالب معين من الطلاب لهذا يطلق عليه  اسم المثيل Instance .

الشكل التالي يوضح فكرة عمل الـ Class والـ Object :


فيما يلي  نظرة سريعة  على بعض المصطلحات المرتبطة بالبرمجة بإسلوب OOP :

  • الأسلوب (Method) : نوع خاص من الوظائف ( الدوال Functions ) يتم تعريفها في الـ Class.(المزيد في مقال الدوال Functions في Python)
  • متغير المثيل (Instance variable) : ويعرف أيضا بـسمة المثيل (Instance Attribute ) متغير يتم تعريفه داخل الدالة وينتمي فقط إلى النسخة الحالية للـ Class بمعنى لكل Object نسخة خاصة من هذا المتغير تحمل بيانات هذا الـ Object. هذا النوع من المتغيرات يمثل الـ Class عند حالة معينة لهذا تعرف هذه المتغيرات بمتغيرات الحالة.
  • متغير الفئة (Class variable) : ويعرف أيضاً بـسمة الفئة (Class Attribute) متغير مشترك بين كافة مثيلات (الـ Objects الخاصة بهذا  الـ Class) (instances of a class). يتم تعريف متغيرات الفئة داخل الـ Class ولكن خارج أي من أساليب (Methods) الـ Class.لا يتم استخدام متغيرات الفئة بشكل متكرر مثل متغيرات المثيل (Instance variable).
  • عنصر البيانات (Data member): أي متغير  سواء متغير فئة Class variable أو متغير حالة Instance variable يحتفظ بالبيانات المرتبطة بالـ Class وكائناتها .
  • السمات (Attributes): هي عناصر البيانات (متغيرات الـ Class و متغيرات الحالة). 
  • الفئة Class: نموذج أولي محدد من قبل المستخدم لكائن(Object) يحدد مجموعة من السمات (Attributes) التي تميز أي كائن أي Class، ويحتوي الـ Class على الأساليب (Methods) التي تصف سلوك الـ Class و يمكن الوصول الى  السمات و الأساليب الخاصة بـالـ Class عبر تدوين النقطة dot notation. . 
  • الكائن (Object): تمثيل فريد لبنية البيانات التي يتم تعريفها بواسطة الـ Class. يشتمل الكائن على قيم  لعناصر البيانات (متغيرات الـ Class ومتغيرات الحالة) والأساليب (Methods). (نسخة من الـ Class ببيانات حقيقية).
  • المثيل (Instance): كل كائن فردي ( An individual object ) من Class معين . بمعنى أي تمثيل لحالة معينة من الـ Class  بواسطة كائن Object يسمى "مثيل Instance" مثلاً الكائن object الذي ينتمي إلى فئة Circle هو مثيل للفئة Circle.

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


أمثلة تطبيقية على Class & Object

وللتعرف أكثر على الـ Class & Object تابع /ي الأمثلة  التطبيقة  التالية عن مفاهيم الـ Class & Objects في لغة بايثون:

إنشاء Class & Object

في المثال التالي سنقوم بإنشاء Class خاص بالموظفين وسنقوم باضافة و دوال للتعامل مع بيانات الـ Objects المختلفة ، ثم سنرى طريقة عمل الـ Objects لهذا الـ Class ونوضح طريقة العمل مع البيانات :


#Create a class and named it "Employees"
class Employees:
  #The frist method in Employees class 
  def SetEmployee(self, name):
    self.name = name
  
  #The second method in Employees class  
  def GetEmployee(self):
    print("The employee name is", self.name)

ولإنشاء Object من الـ Class أعلاه نكتب….


...........
  #Create the first object (instance) to the Employees class  
  employee1 = Employees()
  #calling the Employees class methods by it's instance 
  employee1.SetEmployee("Nora")
  employee1.GetEmployee()
  
  #Create the second instance 
  employee2 = Employees()
  employee2.SetEmployee("Sara")
  employee2.GetEmployee()
...........

في المثال السابق نلاحظ:

  • قمنا بإنشاء الـ Class بواسطة الكلمة المحجوزة class متبوعه بإسم الـ Class. ثم قمنا باضافة الدوال (Methods) للـ Class واستخدمت الكلمة المحجوزة self مع الدوال لتشير إلى الـ Object المستخدم حالياً.
  • ثم قمنا بإنشاء الـ Objects للـ Class، والدوال يتم إنشاؤها داخل الـ Class وتستدعى بواسطة مثيلات (Instances) الـ class (الكائنات  Objects التي إنشائها لهذا الـ Class).

الـ Constructor

المثال التالي يوضح طريقة اضافة الـ Attributes (بيانات الكائن) في الـ Class بواسطة الـ Constructor. و الـ Constructor نوع خاص من الدوال يستخدم بشكل أساسي لتهيئة القيم للـ Attributes فهو كما نعلم يستدعى بشكل تلقائي عند إنشاء الـ Object  (تم شرح الـ Constructor في مقال بعنوان مقدمة في Class & Object للغة #C)، في بايثون يتم استخدام الدالة المدمجة __init__ ( self [,args...] ) كا Constructor لنرى المثال التالي:


class Employees:
  #create the constructor 
  def __init__(self, name):
    self.name = name
  
  def GetEmployees(self):
    print("The employee name is", self.name)
    

مع الـ Constructor يمكن اسناد القيم مباشرة لحظة انشاء الـ Object، بالطريقة التالية..


...........
  #Create the first instance to the Employees class  
  employee1 = Employees("Nora") #calling the constructor
  employee1.GetEmployees()
  
  #Create the second instance 
  employee2 = Employees("Sara")
  employee2.GetEmployees()
...........

في بعض لغات البرمجة يمكن انشاء اكثر من Constructor للـ Class الواحد وهي في هذه الحالة تعمل بمدأ Overload أي أنها تتشابه في الاسم وتختلف في الـ Parameters.المثال التالي يوضح طريقة عمل الـ Constructor في بايثون مع عدد متنوع من البيانات  :


class Employees:
  def __init__(self, name, employee_id, department, salary):
    self.name = name
    self.employee_id = employee_id
    self.department = department
    self.salary = salary
  
  def GetEmployees(self):
    print("Employee Name:", self.name)
    print("Employee ID:", self.employee_id)
    print("Department:", self.department)
    print("Salary:", self.salary)
    
def main():
  #Create the instance to the Employees class
  employee1 = Employees("Nora", 123, "HR", 12000) 
  employee1.GetEmployees()
  
  #create the second instance 
  employee2 = Employees("Sara", 321, "IT", 20000)
  employee2.GetEmployees()

if __name__ == '__main__':
  main()  

نلاحظ ان الكود بهذا الشكل غير مرن حيث لابد من توافر كامل البيانات عند إنشاء الـ Object مما يشكل صعوبة في الأداء والتعديل، فضلاً أن حجم الكود سيكون أكبر مع العمل مع عدد بيانات أكثر. 

أحد الطرق المستخدمة مع هذه الحالة في بايثون هو استخدام "kwargs " مع الـ Constructor مما يجعله  يستقبل البيانات ويحفظها بصيغة قاموس Dictionary  (المزيد في مقال عن البيانات التخطيطية Mapping types في Python القواميس Dictionaries) ويمكننا من اختصار الكود الى الشكل التالي:


class Employees:
  def __init__(self,**kwargs):
    self.Data = kwargs
  
  def GetEmployees(self):
    print("The Employee info:")
    for x,y in self.Data.items():
      print(x ,":", y)
    print("\n")
    
    
def main():
  employee1 = Employees(name ="Nora", employee_id=123, department="HR", salary= 12000) 
  employee1.GetEmployees()
  
  employee2 = Employees(name= "Sara", employee_id =321, department= "IT",salary= 20000)
  employee2.GetEmployees()

if __name__ == '__main__':
  main()
  

كما نلاحظ مع استخدام kwargs تم استقبال البيانات في الـ الClass على شكل قاموس Dictionary لهذا تم التعامل معها بواسطة الدوال الخاصة بالقواميس.


السمات Attributes

تُعرف السمات (Attributes ) بالخصائص أو المتغيرات (Variables) المحددة داخل الـ Class . توفر الـ Attribute معلومات حول نوع البيانات التي تحتوي عليها الفئة (Class). هناك نوعان من السمات في بايثون وهما Instance Attributes و Class Attributes. أوضحنا مفهوم كلاً منهما أعلاه وفيما يلي أهم الفروق:

Instance Attributes Class Attributes
يتم تعريف الـ Instance Attributes داخل الـ Constructor. أو داخل اي داله (__init__() function) يتم تعريفها على مستوى الـ Class اي داخل الـ Class وخارج اي داله حتى الـ Constructor.
يتم الوصول إليها بواسطة الـObject حيث يتم كتابة سم الـ Object ثم نقطة ثم اسم الـ Instance Attribute. يمكن الوصول إليها بواسطة أي من اسم الـ Class أو اسم الـ Object .
القيم للـ Instance Attributes مختلفة من object لآخر. القيم للـ Class Attributes مشتركة بين الـ objects من Class معين.
التعديل على قيم الـ Instance Attributes سؤثر فقط على الـ object الذي تمت عليه عملية التعديل. التعديل على قيم الـ Class Attributes يؤثر على قيمة المتغير في بقية الـ objects من هذا الـ Class.

يوضح المثال الفرق بين الـ Instance Attributes و الـ Class Attributes:


class Employees:
  #Declaring the "employee_count" as class attributes
  employee_count = 0
  
  def __init__(self,**kwargs):
    #The "kwargs" is an instance attribute
    self.Data = kwargs
    
    #Access to class attributes by class name
    Employees.employee_count += 1 #make change on class attribute value
    
  def GetEmployees(self):
    print("The info of Employee",Employees.employee_count,"is:")
    for x,y in self.Data.items():
      print(x ,":", y)
    print("\n")
...........
...........
  #Class attributes can be accessed by both class name and object name.
  print("The number of Employees is:", employee2.employee_count)


الدوال أو الأساليب Methods

في بايثون تعتبر الدوال أو الأساليب(Methods) المحرك الأساسي للبرنامج وخصوصاً التي تعمل بنهج البرمجة الموجهة للكائنات OOP . وتنتمي الأساليب (Methods) إلى كائن Object من فئة class وتستخدم لأداء عمليات محددة. ويمكننا تقسيم الدوال أو الأساليب Methods المستخدمة مع الـ Class & Object في نهج OOP في لغة بايثون إلى ثلاث فئات مختلفة وهي: 


دالة المثيل (Instance Method):

وهي الدوال التي يتم تعريفها في الـ Class بشكل اعتيادي ويتم الوصول لها واستخدامها بواسطة الـ Objects ، يمكن لهذا النوع من الدوال الوصول إلى متغيرات المثيل ( instance Attributes) الخاصة بكائن معين . يمكنها أيضًا الوصول إلى متغير الفئة (class Attributes) لأنه مشترك بين جميع الكائنات. ويتضح هذا النوع من الدوال في الأمثلة السابقة…


دالة الفئة (Class Method)

دالة الفئة Class method في بايثون هي دوال مرتبطة بالـ Class وليس بمثيل (Instance) الـ class . هذه الدوال يمكن استدعاؤها بإسم الـ Class فقط ، وليس بواسطة الـ Objects .و توفر Python طريقتين لإنشاء دالة من نوع Class method :

1-بواسطة الدالة classmethod()

تحتوي لغة Python على دالة مدمجة classmethod() والتي تقوم بتحويل الـ instance method إلى class method ويتم استدعاؤها بالإشارة إلى الـ class فقط وليس الكائن object .كما يوضح المثال التالي :


class Students:
  student_count = 0
  
  def __init__(self,**kwargs):
    self.Data = kwargs
    Students.student_count += 1 
    
  #An instance method
  def StudentCount(self):
    print("Student Number is :", self.student_count)
    
  #A class method
  StCounter = classmethod(StudentCount)
    
def main():
  st1 = Students(Name ="Nora", CoursesNo= 6) 
  st2 = Students(Name= "Sara", CoursesNo= 5)
  st3 = Students(Name= "Reem", CoursesNo= 7)

  #calling the instance method
  st2.StudentCount()
  #clalling the class method 
  Students.StCounter()
if __name__ == '__main__':
  main()
  

بواسطة @classmthod

استخدام @classmthod وهي الطريقة الأمثل لتعريف class method حيث إنها أكثر ملاءمة من إعلان الدالة كا instance method أولاً ثم تحويلها إلى class method. فيما يلي مثال يوضح طريقة استخدام @classmthod للإعلان عن class method:


class Students:
  student_count = 0
  
  def __init__(self,Name , CoursesNo):
    self.Name = Name
    self.CoursesNo = CoursesNo
    Students.student_count += 1 
    
  #A class method
  @classmethod
  def newStudent(cls,Name,CoursesNo):
    return cls(Name,CoursesNo)
  
  #The second class method
  @classmethod
  def StudentCount(cls):
    print( "The students number is:",cls.student_count)
    
def main():
  st1 = Students(Name ="Nora", CoursesNo= 6) 
  st2 = Students(Name= "Sara", CoursesNo= 5)
  st3 = Students(Name= "Reem", CoursesNo= 7)
  
  #using the class method
  st4 = Students.newStudent("Asma",4)
  
  #clalling the class method 
  Students.StudentCount()
  
if __name__ == '__main__':
  main()

ملاحظة : في بايثون، تعني "cls" أن الدالة تنتمي إلى الـ class . بينما تعني "self" أن الدالة مرتبطة بمثيل للفئة (Object). وبالتالي يمكن الوصول إلى عناصر البيانات التي تحمل "cls" من خلال اسم الـ Class. من ناحية أخرى، يمكن الوصول إلى عناصر البيانات الذي يحمل "self" بواسطة مثيل instance للفئة (الكائن Object ).


الدالة الثابتة (Static Method)

في بايثون، الدالة الثابتة static method هي نوع من الدوال التي لا تتطلب لإستدعائها إنشاء object ،هي تشبه إلى حد كبير الـ class method ولكن الاختلاف هو أن الـ static method لا تحتوي على argument إلزامية مثل الإشارة إلى الكائن− self أو الإشارة إلى الفئة − cls. وتُستخدم الـ static methods للوصول إلى عناصر البيانات الثابتة (static ) للـ Class معين. وتوفر Python طريقتين لإنشاء دالة ثابتة Static method وهي:

1-بواسطة الدالة staticmethod()

لإنشاء دالة ثابتة static method باستخدام الدالة المدمجة القياسية المسماة staticmethod().التي تقوم بتحويل الـ instance method إلى static method . كما يوضح المثال التالي:


class Students:
  student_count = 0
  def __init__(self,Name , CoursesNo):
    self.Name = Name
    self.CoursesNo = CoursesNo
    Students.student_count += 1 
  

  def StudentCount():
    print("Student Number is :", Students.student_count)
   #A static method    
  Stcounter = staticmethod(StudentCount)
    
def main():
  st1 = Students(Name ="Nora", CoursesNo= 6) 
  st2 = Students(Name= "Sara", CoursesNo= 5)
  st3 = Students(Name= "Reem", CoursesNo= 7)

  #clalling the static method  vis class 
  Students.Stcounter()
  #calling the static method via object  
  st2.Stcounter()
  
if __name__ == '__main__':
  main()
  .

2- بواسطة @staticmethod

الطريقة الثانية لإنشاء دالة ثابتة static method هي استخدام @staticmethod في Python. عندما نستخدم هذه الطريقة مع أي دالة ، فإنه يعطي اشارة الى الـ compiler إلى أن الدالة المحددة هي static method . ويوضح المثال التالي طريقة استخدام @staticmethod:


class Students:
  student_count = 0
  def __init__(self,Name , CoursesNo):
    self.Name = Name
    self.CoursesNo = CoursesNo
    Students.student_count += 1 
  
   #A static method
  @staticmethod
  def StudentCount():
    print("Student Number is :", Students.student_count)
    
def main():
  st1 = Students(Name ="Nora", CoursesNo= 6) 
  st2 = Students(Name= "Sara", CoursesNo= 5)
  st3 = Students(Name= "Reem", CoursesNo= 7)

  #clalling the static method  vis class 
  Students.StudentCount()
  #calling the static method via object  
  st2.StudentCount()
  
if __name__ == '__main__':
  main()
  

خصائص الدالة ثابتة Static Method
  • نظرًا لأن ال دالة ثابتة static method لا يمكنها الوصول إلى الـ class attributes,، فيمكن استخدامها كدالة مساعدة لأداء المهام التي يتم إعادة استخدامها بشكل متكرر.
  • والدوال الثابتة static methods يمكننا استدعائها باستخدام اسم الفئة class name . وبالتالي، فهي تلغي الاعتماد على الكائنات Objects.
  • الدالة الثابتة static method يمكن التنبؤ بها دائمًا حيث يظل سلوكها دون تغيير بغض النظر عن حالة الـ Class .
  • يمكننا إعلان أي دالة كدالة ثابتة static method لمنع التجاوز overriding.

ملاحظة : غالبًا ما يخلط معظمنا بين دوال الفئة Class methods و الدوال الثابتة Static Methods . تذكر دائمًا، لان كلاهما يتم استدعائها بواسطة اسم الـ class . ولكن لا تتمتع الـ Static Methods بالوصول إلى "cls" وبالتالي لا يمكنها تعديل حالة الـ Class .



الفئات المتداخلة Inner Classes

في بايثون يٌعرف الـ Class المعرف داخل class أخر باسم الفئة الداخلية (Inner Class). في بعض الأحيان، تُسمى الـ inner class أيضًا بالفئة المتداخلة nested class . إذا تم إنشاء object للـ inner class، يمكن أن يستخدم هذا الـ object بواسطة الفئة الأصلية (الخارجية). يصبح كائن الفئة الداخلية أحد سمات (Attributes) الفئة الخارجية. وتلقائيًا ترث الفئة الداخلية سمات الفئة الخارجية دون إنشاء وراثة رسميًا.

بمعنى ان تكون الفئة الخارجية (Outer class)، مثل حاوية للفئة الداخلية (inner class) ، بحيث يمكن ان تصل الـ inner class لكل عناصر الـ Outer class ، وتعامل الـ inner class كعنصر بيانات تابع للـ outer class . وتكون الصيغة العامة للإنشاء inner class كتالي:

class outer:
   def __init__(self):
      pass
   class inner:
      def __init__(self):
         pass

ويوضح المثال التالي كيفية العمل مع الفئات المتداخلة Inner classes في لغة بايثون:


class Organization:
  def __init__(self):
    #The inner classes are attribute 
    self.Fin = self.Finance()
    self.market = self.Marketing()
  
  def organization(self):
    print("This an Organization class")
    
  #An inner class
  class Finance:
    def finance(self):
      print("This is Finance Department")
  
  #An inner class  
  class Marketing:
    def marketing(self):
      print("This is Marketing Department")
    

def main():
  # instance of OuterClass
  Org = Organization()
  Org.organization()
  
  #Instance of innerClasses
  Fin = Org.Fin
  Fin.finance()
  
  market = Org.market
  market.marketing()
  
if __name__ == '__main__':
  main()

وفي بايثون يمكن تقسيم الـ inner classes الى :

فئة داخلية متعددة Multiple Inner Class:

في الفئات الداخلية المتعددة، (Multiple Inner Class) يحتوي الـ class الخارجي الواحد على أكثر من class داخلي. ويعمل كل class داخلي بشكل مستقل ولكن يمكن أن يتفاعل مع الـ Outer class. تماما كما في المثال أعلاه .

فئة داخلية متعددة المستويات Multilevel Inner Class:

يشير إلى فئة داخلية تحتوي على فئة داخلية أخرى، بحيث تنشئ مستويات متعددة من الفئات المتداخلة. والشكل التالي يوضح الفرق بين الأشكال التي يمكن أن يكون عليها الـ inner class :

وفي المثال تطبيق لطريقة العمل مع Multilevel Inner Class:


class Organization:
  def __init__(self):
    self.Fin = self.Finance()
    
  def organization(self):
    print("This an Organization class")
    
  #An inner class
  class Finance:
    def __init__(self):
      self.count = self.Accounting()
    def finance(self):
      print("This is Finance Department")
  
  #An inner class inside the Finance class
    class Accounting:
      def accounting(self):
        print("This is Accounting Section")
    

def main():
  # instance of Outer Class
  Org = Organization()
  Org.organization()
  
  #Instance of inner Classes
  Fin = Org.Fin
  Fin.finance()
  
  #Instance of inner - inner Classes
  count = Fin.count
  count.accounting()
  

if __name__ == '__main__':
  main()

جميع الأمثلة لهذا المقال متوفرة على الرابط هنـا.


من خلال فهم الـ OOP في Python، يمكن للمطورين بناء تطبيقات قوية وقابلة للصيانة بشكل أفضل، وزيادة إعادة استخدام الكود وفهمها بشكل أفضل. قمنا في هذا المقال باستعراض المفاهيم الأساسية لهذا الأسلوب وهما Class & Object مع أمثلة تطبيقية توضح كيفية استخدامهما في Python بطريقة بسيطة وفعالة….

إرسال تعليق

فضلاً اترك تعليق

أحدث أقدم

نموذج الاتصال