设为首页 收藏本站
查看: 2185|回复: 0

[经验分享] Python中的父子类之间的继承关系

[复制链接]
累计签到:1 天
连续签到:1 天
发表于 2014-7-11 09:29:25 | 显示全部楼层 |阅读模式
Python作为一个面向对象的脚本语言。这样就很类似于C++ 和JAVA这样的类关系。但是又有自己的特点。在
使用父类的所有函数时。一定要在子类申明有效之后。不然。即使在代码中你看到自己的已经对父类在子类中
进行了实例化或声明,但是这个时候是无效声明。因为你在使用父类之后进行了声明。
会提示您类似于这样的错误提示:
NameError: global name 'mainwin' is not defined。。。。。。
但是你又看到自己的函数中确实进行了对mainwin 的申明。
这就是因为在申明的时候和使用的先后.
每个类可以拥有一个或者多个父类,它们从父类那里继承了属性和方法。如果一个方法在子类的实例中被调用,
或者一个属性在子类的实例中被访问,但是该方法或属性在子类中并不存在,那么就会自动的去其父类中进行查找。
继承父类后,就能调用父类方法和访问父类属性,而要完成整个集成过程,子类是需要调用的构造函数的
子类不显式调用父类的构造方法,而父类构造函数初始化了一些属性,就会出现问题

如果子类和父类都有构造函数,子类其实是重写了父类的构造函数,如果不显式调用父类构造函数,父类的构造函数就不会被执行,
导致子类实例访问父类初始化方法中初始的变量就会出现问题。
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

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

b.funca()

bbb
function b : bbb
Traceback (most recent call last):
  File "D:\workbench\python\MyPythonProject\test\study\overwrite_method.py", line 19, in <module>
    print b.funca()
  File "D:\workbench\python\MyPythonProject\test\study\overwrite_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)更直观一些。


运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-22006-1-1.html 上篇帖子: Python 抓取网页学习系列之一(网页编码格式) 下篇帖子: Python登录模块Demo示例
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表