Python 继承

文章讲述了Python中如何通过继承避免重复代码,特别是在子类的`__init__`方法中使用`super().__init__()`调用父类构造函数,以及如何保持子类属性和父类方法的一致性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

现在创造一个cat实例,由于子类没有自己的构造函数,就会调用到父类的构造函数。

class Mammal:
    def __init__(self,name,sex):
        self.name = name
        self.sex = sex
        self.num_eyes = 2

    def breath(self):
        print(self.name + "在呼吸...")

    def poop(self):
        print(self.name + "在拉屎")

class Human(Mammal):
    def read(self):
        print(self.name + "在阅读")

class Cat(Mammal):
    def scratch_sofa(self):
        print(self.name + "在抓沙发...")
    def poop(self):
        print(self.name + "在猫砂上拉屎...")

cat1 = Cat("Jojo","男")
cat1.poop()

如果子类有自己的拉屎方法的话,就会调用自己的。这背后的逻辑是:

优先看所属的类有没有该方法,没有的话往上找父类的同名方法用。

但如果像这样,给子类写init方法,那创建子类实例时,就会优先调用子类的构造函数。导致实例只有has_tail属性。

class Cat(Mammal):
    def __init__(self):
        self.has_tail = True
    def scratch_sofa(self):
        print(self.name + "在抓沙发...")

一个方法是在子类__init__里面,把name、sex等属性也全部都写上,但这就会造成重复代码,更优雅的做法是,用super()这个方法,super()会返回当前类的父类,所以在子类__init__方法里面,写super().__init__(),这就会调用父类的构造函数。那么子类也会有姓名、性别、眼睛数量的属性了。所以通过继承我们成功把重复的属性和方法代码行给移除了,并且在实例化、获得属性、调用方法时和之前都一模一样,没有任何区别。

class Cat(Mammal):
    def __init__(self,name,sex):
        super().__init__(name,sex)
        self.has_tail = True
    def scratch_sofa(self):
        print(self.name + "在抓沙发...")

cat1 = Cat("Jojo","男")
print(cat1.name)

在子类下面调用super方法,super()会返回当前类的父类,所以在子类的__init__方法里,写super().__init__()就会调用父类的构造函数。子类也会有姓名、性别、眼睛和数量的属性了。通过继承,我们成功把重复的属性和方法代码行给移除了。

<think>好的,用户想了解Python继承的用法和示例,以及解决继承相关的问题。首先,我需要回忆Python继承的基本概念。Python支持类的继承,允许子类继承父类的属性和方法。用户可能想知道如何定义子类,如何调用父类的方法,以及如何处理多继承等问题。 首先,我应该从基础开始,比如如何创建一个子类,使用super()函数调用父类的方法。然后可能需要提到方法重写,这是继承中的一个重要概念,子类可以覆盖父类的方法。此外,Python支持多继承,这可能引发一些问题,比如菱形继承问题,需要解释MRO(方法解析顺序)如何工作。 接下来,考虑用户的实际应用场景,可能需要提供一些代码示例。比如,定义一个父类Animal,子类Dog继承它,并重写某些方法。同时,展示super()的使用,特别是在初始化父类时的应用。另外,多继承的例子,比如一个子类继承自两个父类,说明方法解析顺序。 用户可能遇到的问题包括父类方法的正确调用,多继承时的冲突,以及如何管理不同父类的初始化。需要解释MRO的动态性,以及如何查看类的MRO列表。此外,访问控制的问题,比如Python中没有真正的私有变量,但通过命名约定(如单下划线和双下划线)来实现封装,这可能和继承有关联。 还需要考虑引用中的内容,引用[2]和[3]提到访问修饰符,虽然Python没有严格的访问控制,但通过下划线来约定。这可能影响子类对父类属性和方法的访问,需要说明子类如何处理这些情况。 最后,生成相关问题,帮助用户进一步学习,比如继承中的super()作用,多继承的问题,方法重写,访问控制等。确保问题覆盖用户可能遇到的常见疑问,并引导他们深入理解Python继承的机制。</think>### Python继承的用法与示例 #### 一、继承基础 Python通过`class 子类名(父类名)`语法实现继承。子类自动获得父类的属性和方法: ```python class Animal: def __init__(self, name): self.name = name def speak(self): return "未定义叫声" class Dog(Animal): def speak(self): # 方法重写 return f"{self.name}说: 汪汪!" ``` #### 二、super()函数 使用`super()`调用父类方法,特别是在处理__init__时: ```python class Cat(Animal): def __init__(self, name, breed): super().__init__(name) # 调用父类初始化 self.breed = breed def speak(self): base_sound = super().speak() return f"{base_sound}(实际是喵喵)" ``` #### 三、多继承与MRO Python支持多继承,方法解析顺序(MRO)遵循C3算法: ```python class A: def test(self): print("A") class B(A): def test(self): print("B") class C(A): def test(self): print("C") class D(B, C): pass D().test() # 输出B,遵循D->B->C->A顺序 print(D.mro()) # 查看方法解析顺序 ``` #### 四、访问控制 Python通过命名约定实现访问控制[^2][^3]: ```python class Parent: def __init__(self): self.public = 1 self._protected = 2 self.__private = 3 # 会重命名为_Parent__private class Child(Parent): def show(self): print(self.public) # 可访问 print(self._protected) # 约定上不直接访问 # print(self.__private) # 会报错 ``` #### 五、抽象基类 使用abc模块定义抽象方法: ```python from abc import ABC, abstractmethod class Shape(ABC): @abstractmethod def area(self): pass class Circle(Shape): def __init__(self, radius): self.radius = radius def area(self): return 3.14 * self.radius ** 2 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值