python 将有参函数转化为无参函数
一、说明
这个过程类似“打包”,在一些游戏开发场景中,常常有将函数延后执行或者需要将函数作为回调入参,这时函数需要带参就不好处理了。例如:
import Functor sUser = ‘用户名‘ iAcct = 123 sPass = ‘******‘ def notify(sUser, bSuc): if bSuc: print sUser + ‘成功登录‘ ‘‘‘login(iAcct, sPass, pfunCB)负责执行登录过程, 在结束时调用pfunCB(bSuc)通知登录结果 ‘‘‘ pfunCB = Functor(notify, sUser) login(iAcct, sPass, pfunCB)
上面是一个登录过程,Functor就是要实现的打包器,将notify(sUser, bSuc)打包为只需一个入参的pfunCB(bSuc)
二、python的方法相关知识
1.python的方法是对象
方法的调用实质上是调用方法实例的__call__(),
因此,把对象变为可调用项,只需添加__call__()。例:
class Fun(): def __init__(self): pass def __call__(self, obj): print obj a = Fun() a(67890) # 等同a.__call__(67890)
2.实例方法的关键属性
对于实例方法的函数对象,我们可以通过__func__或im_func属性获取原始的函数对象。
下面的例子中,Functor还引用了实例方法的实例对象,因为当实例被回收时,我们不希望再调用该实例的方法,需要兼容这种情况。当然,更合理的做法是在代码中规避这种用法,不合理的使用常常导致bug的产生。
可以通过__self__或im_self获取实例方法的实例对象,类方法也可以这样获取到类实例(注意类方法的类实例不是im_class属性)。
三、Functor的实现
=======================Functor.py=======================
# -*- coding:utf-8 -*- import types import weakref class Functor(): def __init__(self, _fn, *args, **kwargs): if not callable(_fn): raise Exception(‘object is unable to call‘) if isinstance(_fn, types.MethodType): # 实例方法,类方法 # 弱引用实例对象,使Functor不影响对象的生命周期 self.m_wrObj = weakref.ref(_fn.__self__) self.m_Func = _fn.im_func self.m_bIsMethodType = True else: # 非实例方法 self.m_wrObj = None self.m_Func = _fn self.m_bIsMethodType = False self.m_args = args self.m_kwargs = kwargs def __call__(self, *args, **kwargs): curkwargs = dict(self.m_kwargs) curkwargs.update(kwargs) if not self.m_bIsMethodType: return self.m_Func(*(self.m_args + args), **kwargs) else: obj = self.m_wrObj() if obj is None: return None return self.m_Func(*((obj,) + self.m_args + args), **kwargs) #test len(‘abc‘) len_no_arg = Functor(len, ‘abc‘) print len_no_arg()