首页 > 编程语言 >第10篇:从入门到精通:深入理解Python继承与多态的概念及应用

第10篇:从入门到精通:深入理解Python继承与多态的概念及应用

时间:2025-01-19 09:03:24浏览次数:3  
标签:10 Python self 多态 print 父类 super class def

第10篇:继承与多态

内容简介

本篇文章将深入探讨Python中的继承与多态概念。您将学习如何通过类的继承实现代码的重用,掌握方法重写的技巧,了解如何使用super()函数调用父类的方法,并探索多态的实现与应用。通过丰富的代码示例,您将能够熟练运用继承与多态,提升您的面向对象编程(OOP)能力和代码的灵活性。


目录

  1. 继承与多态概述
    • 什么是继承
    • 什么是多态
    • 面向对象编程中继承与多态的作用
  2. 类的继承
    • 父类与子类
    • 继承的语法
    • 单继承与多继承
  3. 方法重写
    • 什么是方法重写
    • 重写方法的规则
    • 使用super()函数调用父类的方法
  4. super() 函数的使用
    • super()的作用
    • super()访问父类的属性和方法
    • 示例
  5. 多态的实现与应用
    • 多态的定义
    • 如何在Python中实现多态
    • 多态的实际应用示例
  6. 示例代码
    • 继承示例
    • 方法重写示例
    • 使用super()函数的示例
    • 多态示例
  7. 常见问题及解决方法
    • 问题1:如何实现多重继承?
    • 问题2:如何避免方法重写中的错误?
    • 问题3:如何使用super()与多继承?
    • 问题4:多态与鸭子类型的区别?
  8. 总结

继承与多态概述

什么是继承

**继承(Inheritance)**是面向对象编程(OOP)中的一个基本概念,它允许一个类(子类)继承另一个类(父类)的属性和方法。通过继承,子类可以复用父类的代码,并根据需要进行扩展或修改。

什么是多态

**多态(Polymorphism)**是指不同类的对象可以通过相同的接口调用各自的方法,实现同一操作在不同对象上的不同表现。多态提高了代码的灵活性和可扩展性,使得程序更具通用性。

面向对象编程中继承与多态的作用

  • 代码重用:通过继承,子类可以复用父类的代码,减少重复劳动。
  • 扩展性:子类可以在继承父类的基础上增加新的属性和方法,满足不同需求。
  • 灵活性:多态允许程序在运行时决定调用哪个类的方法,增强了代码的灵活性和可维护性。
  • 模块化:通过将相关功能封装在类中,代码结构更加清晰,易于管理和维护。

类的继承

父类与子类

  • 父类(基类、超类):被继承的类,定义了子类共享的属性和方法。
  • 子类(派生类):继承父类的类,可以复用和扩展父类的功能。

继承的语法

在Python中,通过在类定义时在括号中指定父类,实现继承。语法如下:

class ParentClass:
    # 父类的属性和方法
    pass

class ChildClass(ParentClass):
    # 子类的属性和方法
    pass

单继承与多继承

  • 单继承:一个子类只能继承一个父类。

    示例

    class Animal:
        def eat(self):
            print("Animal eats")
    
    class Dog(Animal):
        def bark(self):
            print("Dog barks")
    
    dog = Dog()
    dog.eat()  # 输出: Animal eats
    dog.bark() # 输出: Dog barks
    
  • 多继承:一个子类可以继承多个父类,从而结合多个类的属性和方法。

    示例

    class Flyer:
        def fly(self):
            print("Flying")
    
    class Swimmer:
        def swim(self):
            print("Swimming")
    
    class Duck(Flyer, Swimmer):
        def quack(self):
            print("Quacking")
    
    duck = Duck()
    duck.fly()   # 输出: Flying
    duck.swim()  # 输出: Swimming
    duck.quack() # 输出: Quacking
    

方法重写

什么是方法重写

**方法重写(Method Overriding)**是指子类重新定义父类中已经存在的方法,以改变或扩展其行为。通过方法重写,子类可以根据自身需求调整父类的方法实现。

重写方法的规则

  • 子类中定义的方法名、参数列表应与父类中被重写的方法相同。
  • 子类的方法可以调用父类的方法,以复用父类的部分功能。
  • 重写的方法可以完全替代父类的方法,也可以在父类方法的基础上进行扩展。

使用super()函数调用父类的方法

super()函数用于调用父类的方法,允许子类在重写方法时复用父类的方法实现。使用super()可以避免硬编码父类名称,提高代码的可维护性。

示例

class Parent:
    def greet(self):
        print("Hello from Parent")

class Child(Parent):
    def greet(self):
        super().greet()  # 调用父类的方法
        print("Hello from Child")

child = Child()
child.greet()
# 输出:
# Hello from Parent
# Hello from Child

super() 函数的使用

super()的作用

super()函数返回父类的一个代理对象,允许子类调用父类的方法。它主要用于:

  • 调用被子类重写的父类方法。
  • 解决多继承中的父类调用问题,遵循方法解析顺序(MRO)。

super()访问父类的属性和方法

通过super(),子类可以访问父类的属性和方法,而无需明确指定父类的名称。这使得代码更加灵活,特别是在多继承的情况下。

示例

class Vehicle:
    def __init__(self, brand):
        self.brand = brand

    def start_engine(self):
        print(f"{self.brand} engine started.")

class Car(Vehicle):
    def __init__(self, brand, model):
        super().__init__(brand)  # 调用父类的构造函数
        self.model = model

    def start_engine(self):
        super().start_engine()  # 调用父类的方法
        print(f"{self.brand} {self.model} is ready to go!")

car = Car("Toyota", "Corolla")
car.start_engine()
# 输出:
# Toyota engine started.
# Toyota Corolla is ready to go!

示例

class Employee:
    def __init__(self, name, salary):
        self.name = name
        self.salary = salary

    def get_details(self):
        return f"Name: {self.name}, Salary: {self.salary}"

class Manager(Employee):
    def __init__(self, name, salary, department):
        super().__init__(name, salary)  # 调用父类的构造函数
        self.department = department

    def get_details(self):
        details = super().get_details()  # 调用父类的方法
        return f"{details}, Department: {self.department}"

manager = Manager("Alice", 90000, "HR")
print(manager.get_details())
# 输出: Name: Alice, Salary: 90000, Department: HR

多态的实现与应用

多态的定义

**多态(Polymorphism)**指的是不同类的对象可以通过相同的接口调用各自的方法,实现相同操作在不同对象上的不同表现。多态性提高了代码的灵活性和可扩展性。

如何在Python中实现多态

在Python中,多态主要通过以下方式实现:

  • 方法重写:子类重写父类的方法,实现不同的行为。
  • 鸭子类型(Duck Typing):不关注对象的类型,只关注对象是否具备所需的方法或属性。

多态的实际应用示例

示例1:方法重写实现多态

class Animal:
    def speak(self):
        pass

class Dog(Animal):
    def speak(self):
        return "Woof!"

class Cat(Animal):
    def speak(self):
        return "Meow!"

class Bird(Animal):
    def speak(self):
        return "Chirp!"

def animal_sound(animal):
    print(animal.speak())

# 创建不同的动物对象
dog = Dog()
cat = Cat()
bird = Bird()

# 多态应用
animal_sound(dog)   # 输出: Woof!
animal_sound(cat)   # 输出: Meow!
animal_sound(bird)  # 输出: Chirp!

示例2:鸭子类型实现多态

class Writer:
    def write(self):
        pass

class PythonWriter(Writer):
    def write(self):
        print("Writing Python code.")

class JavaWriter(Writer):
    def write(self):
        print("Writing Java code.")

class CSharpWriter(Writer):
    def write(self):
        print("Writing C# code.")

def perform_write(writer):
    writer.write()

# 创建不同的写作者对象
python_writer = PythonWriter()
java_writer = JavaWriter()
csharp_writer = CSharpWriter()

# 通过相同的接口调用不同的方法
perform_write(python_writer)  # 输出: Writing Python code.
perform_write(java_writer)    # 输出: Writing Java code.
perform_write(csharp_writer)  # 输出: Writing C# code.

示例代码

继承示例

以下示例展示了如何通过继承创建子类,并复用父类的属性和方法。

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def introduce(self):
        return f"Hi, I'm {self.name} and I'm {self.age} years old."

class Student(Person):
    def __init__(self, name, age, student_id):
        super().__init__(name, age)  # 调用父类的构造函数
        self.student_id = student_id

    def introduce(self):
        base_introduction = super().introduce()
        return f"{base_introduction} My student ID is {self.student_id}."

# 创建对象实例
student = Student("Tom", 20, "S12345")
print(student.introduce())
# 输出: Hi, I'm Tom and I'm 20 years old. My student ID is S12345.

方法重写示例

以下示例展示了子类如何重写父类的方法,以实现不同的行为。

class Vehicle:
    def move(self):
        print("Vehicle is moving.")

class Car(Vehicle):
    def move(self):
        print("Car is driving on the road.")

class Boat(Vehicle):
    def move(self):
        print("Boat is sailing on the water.")

# 创建对象实例
car = Car()
boat = Boat()

# 调用重写的方法
car.move()   # 输出: Car is driving on the road.
boat.move()  # 输出: Boat is sailing on the water.

使用super()函数的示例

以下示例展示了如何使用super()函数调用父类的方法和构造函数。

class Employee:
    def __init__(self, name, salary):
        self.name = name
        self.salary = salary

    def work(self):
        print(f"{self.name} is working.")

class Manager(Employee):
    def __init__(self, name, salary, department):
        super().__init__(name, salary)  # 调用父类的构造函数
        self.department = department

    def work(self):
        super().work()  # 调用父类的方法
        print(f"{self.name} is managing the {self.department} department.")

# 创建对象实例
manager = Manager("Alice", 80000, "Sales")
manager.work()
# 输出:
# Alice is working.
# Alice is managing the Sales department.

多态示例

以下示例展示了多态的实现,通过相同的接口调用不同类的方法。

class Shape:
    def area(self):
        pass

class Rectangle(Shape):
    def __init__(self, width, height):
        self.width = width
        self.height = height

    def area(self):
        return self.width * self.height

class Circle(Shape):
    def __init__(self, radius):
        self.radius = radius

    def area(self):
        return 3.1416 * self.radius ** 2

def print_area(shape):
    print(f"Area: {shape.area()}")

# 创建不同形状的对象
rectangle = Rectangle(4, 5)
circle = Circle(3)

# 多态应用
print_area(rectangle)  # 输出: Area: 20
print_area(circle)     # 输出: Area: 28.2744

常见问题及解决方法

问题1:如何实现多重继承?

原因:有时候需要一个子类继承多个父类,以结合多个类的属性和方法。

解决方法

在类定义时,在括号中列出多个父类,用逗号分隔。需要注意多重继承可能引发方法解析顺序(MRO)的问题。

示例

class Flyer:
    def fly(self):
        print("Flying")

class Swimmer:
    def swim(self):
        print("Swimming")

class Amphibian(Flyer, Swimmer):
    def move(self):
        self.fly()
        self.swim()

# 创建对象实例
amphibian = Amphibian()
amphibian.move()
# 输出:
# Flying
# Swimming

注意事项

  • 避免菱形继承(Diamond Inheritance),即多个父类有共同的父类,可能导致重复调用父类的方法。
  • 使用super()函数遵循MRO,确保父类方法的正确调用顺序。

问题2:如何避免方法重写中的错误?

原因:在子类重写父类方法时,可能因参数不匹配或调用错误导致程序错误。

解决方法

  • 确保子类方法的参数列表与父类方法一致。
  • 使用super()函数正确调用父类的方法。
  • 利用IDE或代码检查工具检测方法签名的一致性。

示例

class Parent:
    def greet(self, message):
        print(f"Parent says: {message}")

class Child(Parent):
    def greet(self, message):
        super().greet(message)  # 正确调用父类的方法
        print(f"Child echoes: {message}")

child = Child()
child.greet("Hello!")
# 输出:
# Parent says: Hello!
# Child echoes: Hello!

问题3:如何使用super()与多继承?

原因:在多继承情况下,super()函数遵循方法解析顺序(MRO),确保父类方法的正确调用。

解决方法

  • 理解类的MRO,可以通过ClassName.__mro__ClassName.mro()查看。
  • 在多继承中,每个类的方法都应使用super()调用下一个类的方法,避免硬编码父类名称。

示例

class A:
    def do_something(self):
        print("A is doing something")
        super().do_something()

class B:
    def do_something(self):
        print("B is doing something")
        super().do_something()

class C(A, B):
    def do_something(self):
        print("C is doing something")
        super().do_something()

class D:
    def do_something(self):
        print("D is doing something")

# 设置D为C的最后一个父类
C.__bases__ += (D,)

# 创建对象实例
c = C()
c.do_something()
# 输出:
# C is doing something
# A is doing something
# B is doing something
# D is doing something

说明

  • C继承自ABAB都继承自D
  • 使用super()确保每个父类的方法都被调用,遵循MRO。

问题4:多态与鸭子类型的区别?

原因:多态和鸭子类型都是实现灵活代码的手段,但概念上有所不同。

解决方法

  • 多态:通过继承和方法重写,实现不同类对象通过相同接口调用各自的方法。
  • 鸭子类型(Duck Typing):不关心对象的具体类型,只关心对象是否具备所需的方法或属性。

示例

# 多态示例
class Animal:
    def speak(self):
        pass

class Dog(Animal):
    def speak(self):
        print("Woof!")

class Cat(Animal):
    def speak(self):
        print("Meow!")

def make_animal_speak(animal):
    animal.speak()

dog = Dog()
cat = Cat()
make_animal_speak(dog)  # 输出: Woof!
make_animal_speak(cat)  # 输出: Meow!

# 鸭子类型示例
class Bird:
    def fly(self):
        print("Bird is flying")

class Airplane:
    def fly(self):
        print("Airplane is flying")

def let_it_fly(flying_object):
    flying_object.fly()

bird = Bird()
airplane = Airplane()
let_it_fly(bird)      # 输出: Bird is flying
let_it_fly(airplane)  # 输出: Airplane is flying

区别

  • 多态依赖于类的继承关系和方法重写。
  • 鸭子类型不依赖于类的继承关系,只关注对象是否具备所需的方法。

示例代码

继承示例

以下示例展示了如何通过继承创建子类,并复用父类的属性和方法。

class Animal:
    def __init__(self, name):
        self.name = name

    def speak(self):
        print(f"{self.name} makes a sound.")

class Dog(Animal):
    def speak(self):
        print(f"{self.name} says woof!")

class Cat(Animal):
    def speak(self):
        print(f"{self.name} says meow!")

# 创建对象实例
dog = Dog("Buddy")
cat = Cat("Whiskers")

# 调用方法
dog.speak()  # 输出: Buddy says woof!
cat.speak()  # 输出: Whiskers says meow!

方法重写示例

以下示例展示了子类如何重写父类的方法,以实现不同的行为。

class Vehicle:
    def move(self):
        print("Vehicle is moving.")

class Car(Vehicle):
    def move(self):
        print("Car is driving on the road.")

class Boat(Vehicle):
    def move(self):
        print("Boat is sailing on the water.")

# 创建对象实例
car = Car()
boat = Boat()

# 调用重写的方法
car.move()   # 输出: Car is driving on the road.
boat.move()  # 输出: Boat is sailing on the water.

使用super()函数的示例

以下示例展示了如何使用super()函数调用父类的方法和构造函数。

class Employee:
    def __init__(self, name, salary):
        self.name = name
        self.salary = salary

    def work(self):
        print(f"{self.name} is working.")

class Manager(Employee):
    def __init__(self, name, salary, department):
        super().__init__(name, salary)  # 调用父类的构造函数
        self.department = department

    def work(self):
        super().work()  # 调用父类的方法
        print(f"{self.name} is managing the {self.department} department.")

# 创建对象实例
manager = Manager("Alice", 80000, "Sales")
manager.work()
# 输出:
# Alice is working.
# Alice is managing the Sales department.

多态示例

以下示例展示了多态的实现,通过相同的接口调用不同类的方法。

class Shape:
    def area(self):
        pass

class Rectangle(Shape):
    def __init__(self, width, height):
        self.width = width
        self.height = height

    def area(self):
        return self.width * self.height

class Circle(Shape):
    def __init__(self, radius):
        self.radius = radius

    def area(self):
        return 3.1416 * self.radius ** 2

def print_area(shape):
    print(f"Area: {shape.area()}")

# 创建不同形状的对象
rectangle = Rectangle(4, 5)
circle = Circle(3)

# 多态应用
print_area(rectangle)  # 输出: Area: 20
print_area(circle)     # 输出: Area: 28.2744

常见问题及解决方法

问题1:如何实现多重继承?

原因:有时候需要一个子类继承多个父类,以结合多个类的属性和方法。

解决方法

在类定义时,在括号中列出多个父类,用逗号分隔。需要注意多重继承可能引发方法解析顺序(MRO)的问题。

示例

class Flyer:
    def fly(self):
        print("Flying")

class Swimmer:
    def swim(self):
        print("Swimming")

class Amphibian(Flyer, Swimmer):
    def move(self):
        self.fly()
        self.swim()

# 创建对象实例
amphibian = Amphibian()
amphibian.move()
# 输出:
# Flying
# Swimming

注意事项

  • 避免菱形继承(Diamond Inheritance),即多个父类有共同的父类,可能导致重复调用父类的方法。
  • 使用super()函数遵循MRO,确保父类方法的正确调用顺序。

问题2:如何避免方法重写中的错误?

原因:在子类重写父类方法时,可能因参数不匹配或调用错误导致程序错误。

解决方法

  • 确保子类方法的参数列表与父类方法一致。
  • 使用super()函数正确调用父类的方法。
  • 利用IDE或代码检查工具检测方法签名的一致性。

示例

class Parent:
    def greet(self, message):
        print(f"Parent says: {message}")

class Child(Parent):
    def greet(self, message):
        super().greet(message)  # 正确调用父类的方法
        print(f"Child echoes: {message}")

child = Child()
child.greet("Hello!")
# 输出:
# Parent says: Hello!
# Child echoes: Hello!

问题3:如何使用super()与多继承?

原因:在多继承情况下,super()函数遵循方法解析顺序(MRO),确保父类方法的正确调用。

解决方法

  • 理解类的MRO,可以通过ClassName.__mro__ClassName.mro()查看。
  • 在多继承中,每个类的方法都应使用super()调用下一个类的方法,避免硬编码父类名称。

示例

class A:
    def do_something(self):
        print("A is doing something")
        super().do_something()

class B:
    def do_something(self):
        print("B is doing something")
        super().do_something()

class C(A, B):
    def do_something(self):
        print("C is doing something")
        super().do_something()

class D:
    def do_something(self):
        print("D is doing something")

# 设置D为C的最后一个父类
C.__bases__ += (D,)

# 创建对象实例
c = C()
c.do_something()
# 输出:
# C is doing something
# A is doing something
# B is doing something
# D is doing something

说明

  • C继承自ABAB都继承自D
  • 使用super()确保每个父类的方法都被调用,遵循MRO。

问题4:多态与鸭子类型的区别?

原因:多态和鸭子类型都是实现灵活代码的手段,但概念上有所不同。

解决方法

  • 多态:通过继承和方法重写,实现不同类对象通过相同接口调用各自的方法。
  • 鸭子类型(Duck Typing):不关心对象的类型,只关心对象是否具备所需的方法或属性。

示例

# 多态示例
class Animal:
    def speak(self):
        pass

class Dog(Animal):
    def speak(self):
        print("Woof!")

class Cat(Animal):
    def speak(self):
        print("Meow!")

def make_animal_speak(animal):
    animal.speak()

dog = Dog()
cat = Cat()
make_animal_speak(dog)  # 输出: Woof!
make_animal_speak(cat)  # 输出: Meow!

# 鸭子类型示例
class Bird:
    def fly(self):
        print("Bird is flying")

class Airplane:
    def fly(self):
        print("Airplane is flying")

def let_it_fly(flying_object):
    flying_object.fly()

bird = Bird()
airplane = Airplane()
let_it_fly(bird)      # 输出: Bird is flying
let_it_fly(airplane)  # 输出: Airplane is flying

区别

  • 多态依赖于类的继承关系和方法重写。
  • 鸭子类型不依赖于类的继承关系,只关注对象是否具备所需的方法。

示例代码

继承示例

以下示例展示了如何通过继承创建子类,并复用父类的属性和方法。

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def introduce(self):
        return f"Hi, I'm {self.name} and I'm {self.age} years old."

class Student(Person):
    def __init__(self, name, age, student_id):
        super().__init__(name, age)  # 调用父类的构造函数
        self.student_id = student_id

    def introduce(self):
        base_introduction = super().introduce()
        return f"{base_introduction} My student ID is {self.student_id}."

# 创建对象实例
student = Student("Tom", 20, "S12345")
print(student.introduce())
# 输出: Hi, I'm Tom and I'm 20 years old. My student ID is S12345.

方法重写示例

以下示例展示了子类如何重写父类的方法,以实现不同的行为。

class Vehicle:
    def move(self):
        print("Vehicle is moving.")

class Car(Vehicle):
    def move(self):
        print("Car is driving on the road.")

class Boat(Vehicle):
    def move(self):
        print("Boat is sailing on the water.")

# 创建对象实例
car = Car()
boat = Boat()

# 调用重写的方法
car.move()   # 输出: Car is driving on the road.
boat.move()  # 输出: Boat is sailing on the water.

使用super()函数的示例

以下示例展示了如何使用super()函数调用父类的方法和构造函数。

class Employee:
    def __init__(self, name, salary):
        self.name = name
        self.salary = salary

    def work(self):
        print(f"{self.name} is working.")

class Manager(Employee):
    def __init__(self, name, salary, department):
        super().__init__(name, salary)  # 调用父类的构造函数
        self.department = department

    def work(self):
        super().work()  # 调用父类的方法
        print(f"{self.name} is managing the {self.department} department.")

# 创建对象实例
manager = Manager("Alice", 80000, "Sales")
manager.work()
# 输出:
# Alice is working.
# Alice is managing the Sales department.

多态示例

以下示例展示了多态的实现,通过相同的接口调用不同类的方法。

class Shape:
    def area(self):
        pass

class Rectangle(Shape):
    def __init__(self, width, height):
        self.width = width
        self.height = height

    def area(self):
        return self.width * self.height

class Circle(Shape):
    def __init__(self, radius):
        self.radius = radius

    def area(self):
        return 3.1416 * self.radius ** 2

def print_area(shape):
    print(f"Area: {shape.area()}")

# 创建不同形状的对象
rectangle = Rectangle(4, 5)
circle = Circle(3)

# 多态应用
print_area(rectangle)  # 输出: Area: 20
print_area(circle)     # 输出: Area: 28.2744

常见问题及解决方法

问题1:如何实现多重继承?

原因:有时候需要一个子类继承多个父类,以结合多个类的属性和方法。

解决方法

在类定义时,在括号中列出多个父类,用逗号分隔。需要注意多重继承可能引发方法解析顺序(MRO)的问题。

示例

class Flyer:
    def fly(self):
        print("Flying")

class Swimmer:
    def swim(self):
        print("Swimming")

class Amphibian(Flyer, Swimmer):
    def move(self):
        self.fly()
        self.swim()

# 创建对象实例
amphibian = Amphibian()
amphibian.move()
# 输出:
# Flying
# Swimming

注意事项

  • 避免菱形继承(Diamond Inheritance),即多个父类有共同的父类,可能导致重复调用父类的方法。
  • 使用super()函数遵循MRO,确保父类方法的正确调用顺序。

问题2:如何避免方法重写中的错误?

原因:在子类重写父类方法时,可能因参数不匹配或调用错误导致程序错误。

解决方法

  • 确保子类方法的参数列表与父类方法一致。
  • 使用super()函数正确调用父类的方法。
  • 利用IDE或代码检查工具检测方法签名的一致性。

示例

class Parent:
    def greet(self, message):
        print(f"Parent says: {message}")

class Child(Parent):
    def greet(self, message):
        super().greet(message)  # 正确调用父类的方法
        print(f"Child echoes: {message}")

child = Child()
child.greet("Hello!")
# 输出:
# Parent says: Hello!
# Child echoes: Hello!

问题3:如何使用super()与多继承?

原因:在多继承情况下,super()函数遵循方法解析顺序(MRO),确保父类方法的正确调用。

解决方法

  • 理解类的MRO,可以通过ClassName.__mro__ClassName.mro()查看。
  • 在多继承中,每个类的方法都应使用super()调用下一个类的方法,避免硬编码父类名称。

示例

class A:
    def do_something(self):
        print("A is doing something")
        super().do_something()

class B:
    def do_something(self):
        print("B is doing something")
        super().do_something()

class C(A, B):
    def do_something(self):
        print("C is doing something")
        super().do_something()

class D:
    def do_something(self):
        print("D is doing something")

# 设置D为C的最后一个父类
C.__bases__ += (D,)

# 创建对象实例
c = C()
c.do_something()
# 输出:
# C is doing something
# A is doing something
# B is doing something
# D is doing something

说明

  • C继承自ABAB都继承自D
  • 使用super()确保每个父类的方法都被调用,遵循MRO。

问题4:多态与鸭子类型的区别?

原因:多态和鸭子类型都是实现灵活代码的手段,但概念上有所不同。

解决方法

  • 多态:通过继承和方法重写,实现不同类对象通过相同接口调用各自的方法。
  • 鸭子类型(Duck Typing):不关心对象的类型,只关心对象是否具备所需的方法或属性。

示例

# 多态示例
class Animal:
    def speak(self):
        pass

class Dog(Animal):
    def speak(self):
        print("Woof!")

class Cat(Animal):
    def speak(self):
        print("Meow!")

def make_animal_speak(animal):
    animal.speak()

dog = Dog()
cat = Cat()
make_animal_speak(dog)  # 输出: Woof!
make_animal_speak(cat)  # 输出: Meow!

# 鸭子类型示例
class Bird:
    def fly(self):
        print("Bird is flying")

class Airplane:
    def fly(self):
        print("Airplane is flying")

def let_it_fly(flying_object):
    flying_object.fly()

bird = Bird()
airplane = Airplane()
let_it_fly(bird)      # 输出: Bird is flying
let_it_fly(airplane)  # 输出: Airplane is flying

区别

  • 多态依赖于类的继承关系和方法重写。
  • 鸭子类型不依赖于类的继承关系,只关注对象是否具备所需的方法。

总结

在本篇文章中,我们深入探讨了Python中的继承与多态。通过理解继承的基本概念,学习如何通过类的继承实现代码的重用,掌握方法重写的技巧,了解如何使用super()函数调用父类的方法,并探索多态的实现与应用,您已经具备了面向对象编程中继承与多态的基础知识。继承与多态不仅能帮助您编写更灵活、更可维护的代码,还能提升代码的复用性和扩展性。

学习建议

  1. 实践继承与多态项目:通过实际项目,如开发图形绘制应用、管理系统等,巩固所学知识。
  2. 深入学习多继承与MRO:理解多继承的工作原理和方法解析顺序(MRO),避免潜在的问题。
  3. 探索高级OOP概念:如抽象类、接口、装饰器等,扩展您的编程技能。
  4. 优化代码设计:学习设计模式(如单例模式、工厂模式、观察者模式),提高代码的设计质量。
  5. 编写文档与测试:为类和方法编写清晰的文档和单元测试,确保代码的可靠性和可维护性。
  6. 参与社区与开源项目:通过参与开源项目,学习他人的代码风格和最佳实践,提升编程能力。
  7. 阅读相关书籍和文档:如《Python编程:从入门到实践》、《面向对象编程与设计模式》,系统性地提升OOP能力。

接下来的系列文章将继续深入探讨Python的异常处理与调试技巧,帮助您进一步掌握Python编程的核心概念和技巧。保持学习的热情,持续实践,您将逐步成为一名优秀的Python开发者!


如果您有任何问题或需要进一步的帮助,请随时在评论区留言或联系相关技术社区。

标签:10,Python,self,多态,print,父类,super,class,def
From: https://blog.csdn.net/martian665/article/details/145063456

相关文章

  • 【行空板K10】第三方库在行空板K10显示不可用怎么办?以CodeBlock为例
    目录引言问题用户库的修改测试 本文首发与DFRobot论坛:第三方库在行空板K10显示不可用怎么办?以CodeBlock为例DF创客社区https://mc.dfrobot.com.cn/thread-323692-1-1.html引言试用了一段时间Mind+图形化编程和行空板K10,感觉非常好用。官方的库基本都已经适配行空板K......
  • python-45-python代码的加速运行优化方式
    文章目录1基础策略和原则1.1常见的优化策略1.2基本的优化原则1.3避免全局变量2避免模块和函数属性访问2.1不推荐写法【32s】2.2消除属性访问(优化1)【28s】2.3局部变量加速(优化2)【24s】2.4局部继续加速(优化3)【22s】2.5减少内层for循环计算......
  • 使用PythonDEAP库实现简单遗传算法
    ​本人博客食用体验更佳哦DEAP(DistributedEvolutionaryAlgorithmsinPython)是一个用于快速原型设计和实验的进化计算框架。它支持多种进化算法,包括遗传算法、遗传编程、进化策略、粒子群优化等。DEAP的设计目标是灵活性和易用性,使得研究人员和开发者能够轻松地实现和测试各......
  • python-leetcode-存在重复元素 II
    219.存在重复元素II-力扣(LeetCode)classSolution:defcontainsNearbyDuplicate(self,nums:List[int],k:int)->bool:seen=set()fori,numinenumerate(nums):ifnuminseen:returnTrue......
  • python-leetcode-最小覆盖子串
    76.最小覆盖子串-力扣(LeetCode)classSolution:defminWindow(self,s:str,t:str)->str:ifnotsornott:return""need={}forcint:need[c]=need.get(c,0)+1windo......
  • 奥运奖牌计数(信息学奥赛一本通-1064)
    【题目描述】2008年北京奥运会,A国的运动员参与了n天的决赛项目(1≤n≤17)。现在要统计一下A国所获得的金、银、铜牌数目及总奖牌数。输入第1行是A国参与决赛项目的天数n,其后n行,每一行是该国某一天获得的金、银、铜牌数目。输出4个整数,为A国所获得的金、银、铜牌总数及总奖牌......
  • 用Python检查Android字符串文件通配符
    #!/usr/bin/envpython3importosimportsysfromdatetimeimportdatetime,timedeltaimportreimportxml.etree.ElementTreeasETiflen(sys.argv)<3:exit()print(datetime.now(),'start')timestamp=int(datetime.now().timestamp())en_t......
  • 统信V20 1070e X86系统编译安装PostgreSQL-13.11版本以及主从构建
    设备信息操作系统版本架构CPU内存备注统信UOSV201070eX864C8G此配置仅做编译安装验证,持续运行或数据量增长大请自行评估资源配置。统信UOSV201070eX864C8G资源包该包包含postgresql-13.11源码包、统信编译postgresql-13.11安装包通过网盘分享的文件:统信postgresq......
  • 4-08动态绑定_多态
    多继承无函数覆盖structBase1 { public: virtualvoidFn_1() { printf("Base1:Fn_1...\n"); } virtualvoidFn_2() { printf("Base1:Fn_2...\n"); } }; structB......
  • 使用Python爬虫将抓取的数据保存到Excel文件
    在进行Python爬虫开发时,数据的存储是非常重要的一环。随着数据分析需求的不断增长,保存和管理大量的数据变得尤为重要。CSV(Comma-SeparatedValues)格式一直是一个常见的存储格式,但在许多应用场景下,Excel文件作为一种更直观、结构化的方式,具有更多的优势,尤其在数据分析与可视......