Python使用装饰器自动调用父类__init__
众所周知,Python中class的构造函数实际是__new__(),但是如果我们执行p1=Point()的时候,不仅会调用Point的__new__方法,而且会调用Point的__init__方法。由于python中所有都是对象,而所有对象的基类都是继承自object,而object实现的__new__基本都满足所有自定义类的要求,所以一般情况下,我们无需重新实现__new__,而只是关注__init__的实现,比如要初始化属性等都放在__init__中。python的这种机制直接导致,如果我们在父类的__init__中实现了一些逻辑,但是子类中又要遵循这些逻辑,就必须显式调用super(cls,self).__init__,在大量的自定义类中来显式调用父类的__init__,让人非常不爽。于是想,有没有一种方法可以让子类能自动隐式调用父类的__init__方法呢?
首先,想到的是python中的装饰器,能用装饰器来实现,最方便不过了,但这里需要解决2个问题:
1)p1=Point(),Point的__new__与__init__都会执行,我们必须将两者分开,能独立执行:创建Point的实例,调用super().__init__,再调用self.__init__
2)装饰器必须返回真实的实例:而不是其它对象
具体实现代码如下:
#使用装饰器自动调用父类的无参__init__方法
def super_init_wrapper(origin_cls):
super_class=origin_cls.mro()[1]
class WrapperClass:
def __new__(cls,*args,**kwargs):
#print('origin class',origin_cls)
instance= object.__new__(origin_cls,*args,**kwargs)
super_class.__init__(instance)
instance.__init__(*args,**kwargs)
return instance
return WrapperClass class Base:
def __init__(self):
self.x=100
print('Base init') @super_init_wrapper
class Instance(Base):
def __init__(self,a,b,c):
self.a=a
self.b=b
self.c=c
print('Instance init') if __name__=='__main__':
ins=Instance('a','b','c')
#ins= super_init_wrapper(Instance)('a','b','c')
print(isinstance(ins,Base))
print(type(ins))
print(ins.x)
import inspect
#kcargs的格式super参数名=self参数名的形式,比如x='{x}'
def auto_super_initial(**kcargs):
def decrorate(self_cls):
base_cls=self_cls.mro()[1]
class DecorateClass:
def __new__(cls,*args,**kwargs):
self=object.__new__(self_cls,*args,**kwargs)
sig=inspect.signature(self.__init__)
init_args=sig.bind_partial(*args,**kwargs).arguments
#构造调用super.__init__的参数
super_args={name:eval(str(value).format(**init_args)) for name,value in kcargs.items()}
base_cls.__init__(self,**super_args)
self_cls.__init__(self,*args,**kwargs)
return self
return DecorateClass
return decrorate class Point:
def __init__(self,x=0,y=0):
self.x=x
self.y=y class Line(Point):
def __init__(self,x=0,y=0,length=0):
super(Triangle,self).__init__(x,y)
self.length=length @auto_super_initial(x='{x}+{y}',y='{y}')
class Rectangle(Point):
def __init__(self,x=0,y=0,width=0,height=0):
self.width=width
self.height=height if __name__=='__main__':
rect=Rectangle(1,10,10,10)
print(rect.x,rect.y) #11 10
Python使用装饰器自动调用父类__init__的更多相关文章
- python 解除装饰器,调用原本函数。
假设fun函数被装饰器装饰了,name调用fun,就不是调用fun本身了,那么如何继续调用本身呢.使用fun_raw = fun.__wrapped__这样使用fun_raw就是调用没被装饰器修饰后的 ...
- Python装饰器的调用过程
在Python学习的过程中,装饰器是比较难理解的一个应用.本人也在学习期间也遇到很多坑,现将装饰器的基本调用过程总结一下. 首先,装饰器用到了“闭包”,而“闭包”是学习装饰器的基础,所以在讲装饰器之前 ...
- python 多个装饰器的调用顺序
python 多个装饰器的调用顺序 一般情况下,在函数中可以使用一个装饰器,但是有时也会有两个或两个以上的装饰器.多个装饰器装饰的顺序是从里到外(就近原则),而调用的顺序是从外到里(就远原则). 原代 ...
- Python:装饰器是如何调用的
应用举例:(1)装饰器 # 装饰器的调用: # 一旦用上装饰器会: # 第一步:调outer函数 # 第二步:被装饰的函数play_game会被当作参数fn给outer # 第三步:最后调用play_ ...
- Python各式装饰器
Python装饰器,分两部分,一是装饰器本身的定义,一是被装饰器对象的定义. 一.函数式装饰器:装饰器本身是一个函数. 1.装饰函数:被装饰对象是一个函数 [1]装饰器无参数: a.被装饰对象无参数: ...
- python基础——装饰器
python基础——装饰器 由于函数也是一个对象,而且函数对象可以被赋值给变量,所以,通过变量也能调用该函数. >>> def now(): ... print('2015-3-25 ...
- 【转】详解Python的装饰器
原文链接:http://python.jobbole.com/86717/ Python中的装饰器是你进入Python大门的一道坎,不管你跨不跨过去它都在那里. 为什么需要装饰器 我们假设你的程序实现 ...
- 理解Python中的装饰器//这篇文章将python的装饰器来龙去脉说的很清楚,故转过来存档
转自:http://www.cnblogs.com/rollenholt/archive/2012/05/02/2479833.html 这篇文章将python的装饰器来龙去脉说的很清楚,故转过来存档 ...
- 详解Python的装饰器
Python中的装饰器是你进入Python大门的一道坎,不管你跨不跨过去它都在那里. 为什么需要装饰器 我们假设你的程序实现了say_hello()和say_goodbye()两个函数. def sa ...
随机推荐
- HTML input 控件
<input type="file" id="file1" onChange="test()"> function test() ...
- JQ 获取下一个元素和获取下一个元素的[指定]子元素
<script type="text/javascript"> $(function () { $("#div1").next().addClass ...
- vue移动端Ui组件 mint-ui 使用指南
1.上啦加载下拉刷新的使用 this.$refs.loadmore.onTopLoaded(); this.$refs.loadmore.onBottomLoaded(); 上啦刷新下拉加载的 动画显 ...
- Python基础(二)数据类型
(一)数字 Python3中的数字类型分为3种,分别是整型,浮点型以及复数. Python2种的数字类型分为4种,分别是整型,长整型,浮点型以及复数. 其中长整型时Python2为应对位数较大的而设置 ...
- PAT_A1125#Chain the Ropes
Source: PAT A1125 Chain the Ropes (25 分) Description: Given some segments of rope, you are supposed ...
- scrapy-redis让redis不止使用db0
废话不多说,直接在 custom_settings 设置即可 代码: class MySpider(RedisSpider): """Spider that reads ...
- C#第十四节课
函数的调用 using System;using System.Collections.Generic;using System.Linq;using System.Text;using System ...
- [luogu 1092] 虫食算 (暴力搜索剪枝)
传送门 Description Input 包含四行. 第一行有一个正整数 (N≤26). 后面的三行,每行有一个由大写字母组成的字符串,分别代表两个加数以及和.这3个字符串左右两端都没有空格,从高位 ...
- #MySQL数据库无法远程访问的问题
在 Ubuntu上装了mysql,因为项目的数据库是mysql,将项目放在tomcat里面webapp下面,一直启动不成功.本来一直以为是jdbc驱动问题,后来发现不是. 感谢!!http://blo ...
- UOJ #131 BZOJ 4199 luogu P2178【NOI2015】品酒大会 (后缀自动机、树形DP)
水是水,但是写出了不少问题,因此写一发博客. https://www.luogu.org/problemnew/show/P2178 https://www.lydsy.com/JudgeOnline ...