Cho đến nay, những chủ đề về lập trình hướng đối tượng trong Python mà chúng ta đã thảo luận là:

–  Phần 1 (lớp, đối tượng và các thành viên của lớp)

– Phần 2 (che giấu dữ liệu và in đối tượng)

Trong bài học này, tính kế thừa sẽ được giới thiệu.

Một trong các lợi thế chính của lập trình hướng đối tượng đó là khả năng tái sử dụng của code. Và kế thừa chính là một trong các cơ chế để đạt được điều đó. Trong kế thừa, một lớp (thường được gọi là lớp cha – super class) sẽ được kế thừa lại bởi một lớp khác (thường gọi là lớp con – subclass). Lớp con này sẽ bổ sung thêm một số thuộc tính cho lớp cha. 

Dưới đây là một đoạn chương trình được viết bằng Python, mô tả cách cài đặt kế thừa trong Python

# -----------------------------------------------------------
#Cafedev.vn - Kênh thông tin IT hàng đầu Việt Nam
#@author cafedevn
#Contact: cafedevn@gmail.com
#Fanpage: https://www.facebook.com/cafedevn
#Instagram: https://instagram.com/cafedevn
#Twitter: https://twitter.com/CafedeVn
#Linkedin: https://www.linkedin.com/in/cafe-dev-407054199/
# -----------------------------------------------------------


# A Python program to demonstrate inheritance  
  
# Base or Super class. Note object in bracket. 
# (Generally, object is made ancestor of all classes) 
# In Python 3.x "class Person" is  
# equivalent to "class Person(object)" 
class Person(object): 
      
    # Constructor 
    def __init__(self, name): 
        self.name = name 
  
    # To get name 
    def getName(self): 
        return self.name 
  
    # To check if this person is employee 
    def isEmployee(self): 
        return False
  
  
# Inherited or Sub class (Note Person in bracket) 
class Employee(Person): 
  
    # Here we return true 
    def isEmployee(self): 
        return True
  
# Driver code 
emp = Person("Geek1")  # An Object of Person 
print(emp.getName(), emp.isEmployee()) 
  
emp = Employee("Geek2") # An Object of Employee 
print(emp.getName(), emp.isEmployee()) 

Kết quả in ra là:

('Geek1', False)
('Geek2', True)

1. Làm thế nào để kiểm tra xem một lớp có phải là lớp con của lớp khác không

Python có cung cấp một hàm có tên là issubclass(), có thể trực tiếp nói cho lập trình viên biết rằng lớp được xét này có phải là lớp con của một lớp khác hay không.


# Python example to check if a class is 
# subclass of another 
  
class Base(object): 
    pass   # Empty Class 
  
class Derived(Base): 
    pass   # Empty Class 
  
# Driver Code 
print(issubclass(Derived, Base)) 
print(issubclass(Base, Derived)) 
  
d = Derived() 
b = Base() 
  
# b is not an instance of Derived 
print(isinstance(b, Derived)) 
  
# But d is an instance of Base 
print(isinstance(d, Base)) 

Kết quả in ra là:

True
False
False
True

2. Object class – lớp đối tượng là gì?

Cũng giống như lớp đối tượng trong Java, bắt đầu từ phiên bản 3.x của Python, đối tượng là gốc rễ của tất cả các lớp.

Trong Python 3.x, “class Test(object)” và “class Test” là như nhau.

Trong Python 2.x, “class Test(object)” sẽ tạo ra một lớp có đối tượng object làm lớp cha (được gọi là new style class), còn “class Test” sẽ tạo ra một old style class – lớp bình thường (không có đối tượng nào làm lớp cha của nó)

3. Python có hỗ trợ đã kế thừa – multiple inheritance không?

Không giống như Java và giống như C++, Python có hỗ trợ đã kế thừa. Chúng ta có thể chỉ định nhiều lớp cha, phân cách nhau bởi dấu phẩy, và tất cả được bao trong một cặp dấu ngoặc tròn () để thực hiện đa kế thừa. 

Dưới đây là ví dụ mô tả cách thức hoạt động của đa kế thừa trong Python


# Python example to show working of multiple  
# inheritance 
class Base1(object): 
    def __init__(self): 
        self.str1 = "Cafedev1"
        print "Base1"
  
class Base2(object): 
    def __init__(self): 
        self.str2 = "Cafedev2"        
        print "Base2"
  
class Derived(Base1, Base2): 
    def __init__(self): 
          
        # Calling constructors of Base1 
        # and Base2 classes 
        Base1.__init__(self) 
        Base2.__init__(self) 
        print "Derived"
          
    def printStrs(self): 
        print(self.str1, self.str2) 
         
  
ob = Derived() 
ob.printStrs() 

Kết quả in ra là:

Cafedev1
Cafedev2
Derived
('Cafedev1', 'Cafedev2')

4. Làm thế nào để truy cập tới các thành viên của lớp cha, tại lớp con?

1. Sử dụng tên lớp cha

Đoạn code ví dụ cho thấy các thành viên của lớp cha có thể được truy cập tới từ bên trong các lớp con, bằng cách sử dụng tên của lớp cha.


# Python example to show that base 
# class members can be accessed in 
# derived class using base class name 
class Base(object): 
  
    # Constructor 
    def __init__(self, x): 
        self.x = x     
  
class Derived(Base): 
  
    # Constructor 
    def __init__(self, x, y): 
        Base.x = x  
        self.y = y 
  
    def printXY(self): 
       
       # print(self.x, self.y) will also work 
       print(Base.x, self.y) 
  
  
# Driver Code 
d = Derived(10, 20) 
d.printXY() 

Kết quả in ra là:

(10, 20)

2. Sử dụng hàm super()

Chúng ta cũng có thể truy cập tới các thành viên của lớp cha bằng cách sử dụng hàm super().

Đoạn code dưới đây sẽ mô tả cách sử dụng hàm super() để truy cập tới các thành viên của lớp cha, từ lớp con

# -----------------------------------------------------------
#Cafedev.vn - Kênh thông tin IT hàng đầu Việt Nam
#@author cafedevn
#Contact: cafedevn@gmail.com
#Fanpage: https://www.facebook.com/cafedevn
#Instagram: https://instagram.com/cafedevn
#Twitter: https://twitter.com/CafedeVn
#Linkedin: https://www.linkedin.com/in/cafe-dev-407054199/
# -----------------------------------------------------------


# Python example to show that base 
# class members can be accessed in 
# derived class using super() 
class Base(object): 
  
    # Constructor 
    def __init__(self, x): 
        self.x = x     
  
class Derived(Base): 
  
    # Constructor 
    def __init__(self, x, y): 
          
        ''' In Python 3.x, "super().__init__(name)" 
            also works''' 
        super(Derived, self).__init__(x) 
        self.y = y 
  
    def printXY(self): 
  
       # Note that Base.x won't work here 
       # because super() is used in constructor 
       print(self.x, self.y) 
  
  
# Driver Code 
d = Derived(10, 20) 
d.printXY() 

Kết quả in ra là:

(10, 20)

Lưu ý rằng, hai phương thức ở trên không hoàn toàn giống nhau. Trong bài học tiếp theo về kế thừa, chúng ta sẽ thảo luận tiếp hai chủ đề sau:

– Cách thức hoạt động của super? Làm thế nào mà việc truy cập tới thành viên (biến thành viên, hàm thành viên) của lớp cha, thông qua super và tên lớp cha lại khác nhau?

– Vấn đề Diamon (kim cương) được giải quyết thế nào trong Python?

Nguồn và Tài liệu tiếng anh tham khảo:

Tài liệu từ cafedev:

Nếu bạn thấy hay và hữu ích, bạn có thể tham gia các kênh sau của cafedev để nhận được nhiều hơn nữa:

Chào thân ái và quyết thắng!

Đăng ký kênh youtube để ủng hộ Cafedev nha các bạn, Thanks you!