python 之 functools模块

functools模块用于高阶函数:作用于或返回其他函数的函数。一般而言,任何可调用对象都可以作为本模块用途的函数来处理。

官方文档已经有了详尽的介绍,这里就不在复述,详情请见:官方文档 。这里主要介绍下 partial。

partial 详解

functools.partial返回的是一个可调用的partial对象,使用方法是partial(func,*args,**kw),func是必须要传入的,而且至少需要一个args或是kw参数。

from functools import partial
def add(a, b, c):
    return a+b+c

p = partial(add, 12)
p(1, 2)
# 15
p(2, 3)
17
p(3, 4)
# 19

  创建一个功能函数,实现三个数的相加,如果其中的一个或是多个参数不变,那么可以使用partial,实例化一个传入了add和12参数的对象,如上图所示,传入两个参数后,得到三个数的和。

partial源码分析:

1.__new__方法部分

    class partial:
        """New function with partial application of the given arguments
        and keywords.
        """
        ###__SLOTS__ 只允许类有此属性,不能动态的添加其他的属性
        __slots__ = "func", "args", "keywords", "__dict__", "__weakref__"
        ###__new__方法生成实例对象
        def __new__(*args, **keywords):
        ###实例化对象时传入参数的限定,不能为空、参数的个数要大于等于2,这就解释了至少需要一个或多个args或是kw,func是一个可调用的对象,是必须要传入的
            if not args:
                raise TypeError("descriptor ‘__new__‘ of partial needs an argument")
            if len(args) < 2:
                raise TypeError("type ‘partial‘ takes at least one argument")
            cls, func, *args = args # args=(cls,func,*args)
            if not callable(func):
                raise TypeError("the first argument must be callable")
            ### 位置参数是以元组的形式传入的
            args = tuple(args)

    ###hasattr这块我也没有咋个明白,不知道可以应用到什么地方,从使用方法来看,传入的函数func要有属性或是方法,如果知道请告知我一下
        if hasattr(func, "func"):
            args = func.args + args
            tmpkw = func.keywords.copy()
            tmpkw.update(keywords)
            keywords = tmpkw
            del tmpkw
            func = func.func
    ###创建一个实例对象本身
        self = super(partial, cls).__new__(cls)
    ###动态的添加属性
        self.func = func
        self.args = args
        self.keywords = keywords
        return self

上面的代码创建了一个实例对象(p=partial(func,*args,**kw)),并给对象本身添加了属性。

2. __call__方法部分

再看可调用的部分,partial实例化的对象是一个可调用的,是因为在partial中写了__call__方法,看源码:

###在使用p()时会自动调用__call__方法
def __call__(*args, **keywords):
    if not args:
        raise TypeError("descriptor ‘__call__‘ of partial needs an argument")
    self, *args = args
###将位置参数和关键字参数分别合在一起,在使用p()的时候只传入了部分的参数,这是为了我们的方便,不重复传入不变的参数,而在__call__方法中会将func所需的参数全部传入
    newkeywords = self.keywords.copy()
    newkeywords.update(keywords)
###*self.args是partial(func,*args,**kw)中的*args
    return self.func(*self.args, *args, **newkeywords)

在使用p(*args,**keywords)的时候,就会自动的调用__call__方法,这就是生成的对象可调用的原因,self是实例化对象本身,*args、**kw是我们传入函数func的参数,但是只是传入了部分参数,这也是partial的作用所在,所以还要将partial(func,*args,**kw)中的位置参数和关键字参数与p(*args,**keywords)一并传入到函数func中来实现函数的功能。官网中解释partial的功能实现相当于:

def partial(func, *args, **keywords):
    def newfunc(*fargs, **fkeywords):
        newkeywords = keywords.copy()
        newkeywords.update(fkeywords)
        return func(*args, *fargs, **newkeywords)
    newfunc.func = func
    newfunc.args = args
    newfunc.keywords = keywords
    return newfunc

当调用partial函数的时候,返回的是newfunc函数对象,让f=partial(add,12),相当了f=newfunc,所以f是一个可调用的对象,因此f(1,2)-->等价于newfunc(1,2)-->将参数整合传入到func中,并返回func实现其功能。

转自:https://blog.csdn.net/Zjack_understands/article/details/80242946

原文地址:https://www.cnblogs.com/zh605929205/p/9940274.html

时间: 2024-10-10 04:12:05

python 之 functools模块的相关文章

Python的functools模块

这个模块提供了3个有趣的函数,这里介绍下其用法. 首先是partial函数,它可以重新绑定函数的可选参数,生成一个callable的partial对象 >>> int('10') # 实际上等同于int('10', base=10)和int('10', 10) 10 >>> int('10', 2) # 实际上是int('10', base=2)的缩写 2 >>> from functools import partial >>> i

python functools模块

functools.partial 作用: functools.partial 通过包装手法,允许我们 "重新定义" 函数签名 用一些默认参数包装一个可调用对象,返回结果是可调用对象,并且可以像原始对象一样对待 冻结部分函数位置函数或关键字参数,简化函数,更少更灵活的函数参数调用 #args/keywords 调用partial时参数 def partial(func, *args, **keywords): def newfunc(*fargs, **fkeywords): newk

Python面向切面编程-语法层面和functools模块

1,Python语法层面对面向切面编程的支持(方法名装饰后改变为log) __author__ = 'Administrator' import time def log(func): def wrapper(*args): start = time.time() func(args) end =time.time() print 'func used time is :', end - start return wrapper @log def reg(args): print 'welcom

python 装饰器和 functools 模块

转自:http://blog.jkey.lu/2013/03/15/python-decorator-and-functools-module/ 什么是装饰器? 在 python 语言里第一次看到装饰器不免让人想到设计模式中的装饰模式——动态地给一个对象添加一些额外的职责,就增加功能来说,装饰模式比生成子类更为灵活. 好吧,python 中的装饰器显然和装饰模式毫无关系.那 python 中的装饰器到底是什么呢? 简而言之,装饰器提供了一种方法,在函数和类定义语句的末尾插入自动运行代码.pyth

python中的functools模块

functools模块可以作用于所有的可以被调用的对象,包括函数 定义了__call__方法的类等 1 functools.cmp_to_key(func) 将比较函数(接受两个参数,通过比较两个参数的大小返回负值,0,或者正数)转换为key function(返回一个值用来比较或者排序的可调用对象), 例如: sorted(iterable, functools.cmp_to_key(locale.strcoll)) def cmp1(n1, n2): return n1 - n2 a = [

Python基础22_模块,collections,time,random,functools

一. 模块 模块: 是把装有特定功能的代码进行归类的结果, 从代码编写的单位来看我们的程序, 从小到大的顺序: 一条代码 < 语句块 < 代码块(函数, 类) < 模块. 我们目前写的所有py文件都是模块 引入模块的方式: 1. import 模块 2. from xxx import 模块 二. collections模块 collections模块主要封装了一些关于集合类的相关操作和一些除了基本数据类型以外的数据集合类型 1. Counter 计数器, 主要用来计数 collecti

Python标准库--functools模块

functools模块:管理函数的工具 partial对象:包装原函数,提供默认值 import functools # 原函数 def myfunc(a, b=2): """Docstring for myfunc().""" print(' called myfunc with:', a, b) return # 输出函数 def show_details(name, f, is_partial=False): print(name) pri

functools模块

functools模块用于高阶函数:作用于或返回其他函数的函数.一般来说,任何可调用的对象都可以作为这个模块的功能来处理. functools模块定义了以下功能: 1.functools.cmp_to_key(func): 将旧式的比较函数转换为关键函数.与接受关键功能的工具一起使用,例如sorted(), min(), max(), heapq.nlargest(), heapq.nsmallest(), itertools.groupby()).该函数主要用作将程序转换为Python 3的转

关于functools模块的wraps装饰器用途

测试环境:Python3.6.2 + win10 +  Pycharm2017.3 装饰器之functools模块的wraps的用途: 首先我们先写一个装饰器 # 探索functools模块wraps装饰器的用途 from functools import wraps def trace(func): """ 装饰器 """ # @wraps(func) def callf(*args, **kwargs): """