Python子类调用父类方法学习笔记

python和其他面向对象语言类似,每个类可以拥有一个或者多个父类,它们从父类那里继承了属性和方法。如果一个方法在子类的实例中被调用,或者一个属性在子类的实例中被访问,但是该方法或属性在子类中并不存在,那么就会自动的去其父类中进行查找。

继承父类后,就能调用父类方法和访问父类属性,而要完成整个集成过程,子类是需要调用的构造函数的。

子类不显式调用父类的构造方法,而父类构造函数初始化了一些属性,就会出现问题
如果子类和父类都有构造函数,子类其实是重写了父类的构造函数,如果不显式调用父类构造函数,父类的构造函数就不会被执行,导致子类实例访问父类初始化方法中初始的变量就会出现问题。

 代码如下 复制代码

class A:
    def __init__(self):
        self.namea="aaa"

    def funca(self):
        print "function a : %s"%self.namea

class B(A):
    def __init__(self):
        self.nameb="bbb"

    def funcb(self):
        print "function b : %s"%self.nameb

b=B()
print b.nameb
b.funcb()

b.funca()

结果:

bbb

 代码如下 复制代码
function b : bbb
Traceback (most recent call last):
  File "D:workbenchpythonMyPythonProjectteststudyoverwrite_method.py", line 19, in <module>
    print b.funca()
  File "D:workbenchpythonMyPythonProjectteststudyoverwrite_method.py", line 6, in funca
    print "function a : %s"%self.namea
AttributeError: B instance has no attribute 'namea'

在子类中,构造函数被重写,但新的构造方法没有任何关于初始化父类的namea属性的代码,为了达到预期的效果,子类的构造方法必须调用其父类的构造方法来进行基本的初始化。有两种方法能达到这个目的:调用超类构造方法的未绑定版本,或者使用super函数。

方法一:调用未绑定的超类构造方法
修改代码,多增一行:

 代码如下 复制代码

class A:
    def __init__(self):
        self.namea="aaa"

    def funca(self):
        print "function a : %s"%self.namea

class B(A):
    def __init__(self):
        #这一行解决了问题
        A.__init__(self)
        self.nameb="bbb"

    def funcb(self):
        print "function b : %s"%self.nameb

b=B()
print b.nameb
b.funcb()

b.funca()

如上有注释的一行解决了该问题,直接使用父类名称调用其构造函数即可。

这种方法叫做调用父类的未绑定的构造方法。在调用一个实例的方法时,该方法的self参数会被自动绑定到实例上(称为绑定方法)。但如果直接调用类的方法(比如A.__init),那么就没有实例会被绑定。这样就可以自由的提供需要的self参数,这种方法称为未绑定unbound方法。

通过将当前的实例作为self参数提供给未绑定方法,B类就能使用其父类构造方法的所有实现,从而namea变量被设置。

方法二:使用super函数
修改代码,这次需要增加在原来代码上增加2行:

 代码如下 复制代码

#父类需要继承object对象
class A(object):
    def __init__(self):
        self.namea="aaa"

    def funca(self):
        print "function a : %s"%self.namea

class B(A):
    def __init__(self):
        #这一行解决问题
        super(B,self).__init__()
        self.nameb="bbb"

    def funcb(self):
        print "function b : %s"%self.nameb

b=B()
print b.nameb
b.funcb()

b.funca()

如上有注释的为新增的代码,其中第一句让类A继承自object类,这样才能使用super函数,因为这是python的“新式类”支持的特性。当前的雷和对象可以作为super函数的参数使用,调用函数返回的对象的任何方法都是调用超类的方法,而不是当前类的方法。

super函数会返回一个super对象,这个对象负责进行方法解析,解析过程其会自动查找所有的父类以及父类的父类。

方法一更直观,方法二可以一次初始化所有超类
super函数比在超累中直接调用未绑定方法更直观,但是其最大的有点是如果子类继承了多个父类,它只需要使用一次super函数就可以。然而如果没有这个需求,直接使用A.__init__(self)更直观一些。

时间: 2024-10-27 05:38:23

Python子类调用父类方法学习笔记的相关文章

Python子类调用父类方法实例程序

python和其他面向对象语言类似,每个类可以拥有一个或者多个父类,它们从父类那里继承了属性和方法.如果一个方法在子类的实例中被调用,或者一个属性在子类的实例中被访问,但是该方法或属性在子类中并不存在,那么就会自动的去其父类中进行查找. 继承父类后,就能调用父类方法和访问父类属性,而要完成整个集成过程,子类是需要调用的构造函数的. 子类不显式调用父类的构造方法,而父类构造函数初始化了一些属性,就会出现问题 如果子类和父类都有构造函数,子类其实是重写了父类的构造函数,如果不显式调用父类构造函数,父

子类调用父类方法的问题

问题描述 子类调用父类方法的问题 父类: package com.example.main;public class Parent { public void print(){ System.out.println(this); }} 子类: package com.example.main;public class Childer extends Parent { public static void main(String[] args) { new Childer().print(); }

java-Java中子类实例可以调用父类方法吗

问题描述 Java中子类实例可以调用父类方法吗 20C 通常情况子类覆写父类方法如果是构造函数,直接用super()如果是普通方法用super.method() 但是如果创建了子类实例,比如A类中覆写了Object的equals()方法A a = new A();a怎么调用Object的equals()方法 A没有显示继承其他类 解决方案 PHP类中子类调用父类方法java中子类继承父类方法及调用规则java中子类,父类的方法调用及属性值 解决方案二: 因为你A类中重写了equals方法了,所以

super-为什么子类重写父类方法之后还要调用父类同名方法

问题描述 为什么子类重写父类方法之后还要调用父类同名方法 为什么子类重写父类方法之后还要调用父类同名方法 比如public class A extends B { public void say ( ) { super.say ( ); } 为什么重写父类方法之后还要在方法体里面用super关键词调用父类的同名方法呢 解决方案 你可以不调用,这个你自定,不是必须 解决方案二: SubClasing用法--子类重写父类方法,父类直接调用实现方式子类调用父类方法子类调用父类方法 解决方案三: 我想到

反编译-子类调用父类的泛型方法,为何不用强转就能得到正确的类型,父类编译的时候不是泛型擦出了吗?

问题描述 子类调用父类的泛型方法,为何不用强转就能得到正确的类型,父类编译的时候不是泛型擦出了吗? 以下面这个简单例子演示: 这是父类 public abstract class GenericFather<T> { public T getT(T t){ return t; } } 通过反编译可以得知编译后泛型擦出,T都用Object代替了 也就是说只要调用父类方法最后肯定返回的是个Object类型 public abstract class GenericFather { public G

Python实现子类调用父类的方法_python

本文实例讲述了Python实现子类调用父类的方法.分享给大家供大家参考.具体实现方法如下: python和其他面向对象语言类似,每个类可以拥有一个或者多个父类,它们从父类那里继承了属性和方法.如果一个方法在子类的实例中被调用,或者一个属性在子类的实例中被访问,但是该方法或属性在子类中并不存在,那么就会自动的去其父类中进行查找. 继承父类后,就能调用父类方法和访问父类属性,而要完成整个集成过程,子类是需要调用的构造函数的. 子类不显式调用父类的构造方法,而父类构造函数初始化了一些属性,就会出现问题

重写-JAVA中子类覆盖了父类的方法,是否还有办法通过子类调用父类同一方法?

问题描述 JAVA中子类覆盖了父类的方法,是否还有办法通过子类调用父类同一方法? JAVA中子类覆盖了父类的方法,是否还有办法通过子类调用父类同一方法? 解决方案 使用super.methodName();调用. Java中super关键字指代的是父类. this关键字指代的是自身. 解决方案二: 用super.方法名调用 解决方案三: 试试super.xxx 解决方案四: 直接用super.方法名调用对应的方法

转型-关于如何通过子类调用父类的父类中的方法

问题描述 关于如何通过子类调用父类的父类中的方法 package access; class A{ public A(){System.out.println("A created!");} public int f(){ System.out.println("call A.f()"); return 1; } } class B extends A{ public int f(){ System.out.println("call B.f()"

请教一下各路大神,C#中关于子类隐藏父类方法或属性的问题

问题描述 请教一下各路大神,C#中关于子类隐藏父类方法或属性的问题 比如说基类A有个方法 print(){}; A的子类B也有个方法print(){}; 那么,在B类中用修饰符new将父类中的该方法隐藏掉有什么意义 不隐藏有什么弊端么 解决方案 有个C#语法的问题吧,子类如果要重写父类的方法的话,virtual-override一定成对的. 子类不重写,而是创建一个属于自己的同名方法,就最好加个new咯. 如果不加new也等于new,但是编译器都会提醒你,加个标识比较好. 区别就是, class