python標准庫--functools.partial


官方相關地址:https://docs.python.org/3.6/library/functools.html

 

一.簡單介紹:
        functools模塊用於高階函數:作用於或返回其他函數的函數。一般而言,任何可調用對象都可以作為本模塊用途的函數來處理。

        functools.partial返回的是一個可調用的partial對象,使用方法是partial(func,*args,**kw),func是必須要傳入的,而且至少需要一個args或是kw參數。

       

  創建一個功能函數,實現三個數的相加,如果其中的一個或是多個參數不變,那么可以使用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實現其功能。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM