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