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()