欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python中的多重裝飾器

 更新時間:2015年04月11日 10:49:27   投稿:junjie  
這篇文章主要介紹了Python中的多重裝飾器,多重裝飾器即多個裝飾器修飾同一個對象,但實際上并非完全如此,本文用實例講解了各種情況,需要的朋友可以參考下

多重裝飾器,即多個裝飾器修飾同一個對象【實際上并非完全如此,且看下文詳解】

1.裝飾器無參數:

復制代碼 代碼如下:

>>> def first(func):
    print '%s() was post to first()'%func.func_name
    def _first(*args,**kw):
        print 'Call the function %s() in _first().'%func.func_name
        return func(*args,**kw)
    return _first


>>> def second(func):
    print '%s() was post to second()'%func.func_name
    def _second(*args,**kw):
        print 'Call the function %s() in _second().'%func.func_name
        return func(*args,**kw)
    return _second


>>> @first
@second
def test():return 'hello world'

test() was post to second()
_second() was post to first()
>>> test()
Call the function _second() in _first().
Call the function test() in _second().
'hello world'
>>>

實際上它是相當于下面的代碼:

復制代碼 代碼如下:

>>> def test():
    return 'hello world'

>>> test=second(test)
test() was post to second()
>>> test
<function _second at 0x000000000316D3C8>
>>> test=first(test)
_second() was post to first()
>>> test
<function _first at 0x000000000316D358>
>>> test()
Call the function _second() in _first().
Call the function test() in _second().
'hello world'
>>>


2.裝飾器有參數:
復制代碼 代碼如下:

>>> def first(printResult=False):
    def _first(func):
        print '%s() was post to _first()'%func.func_name
        def __first(*args,**kw):
            print 'Call the function %s() in __first().'%\
                  func.func_name
            if printResult:
                print func(*args,**kw),'#print in __first().'
            else:
                return func(*args,**kw)
        return __first
    return _first

>>> def second(printResult=False):
    def _second(func):
        print '%s() was post to _second()'%func.func_name
        def __second(*args,**kw):
            print 'Call the function %s() in __second().'%\
                  func.func_name
            if printResult:
                print func(*args,**kw),'#print in __second().'
            else:
                return func(*args,**kw)
        return __second
    return _second

>>> @first(True)
@second(True)
def test():
    return 'hello world'

test() was post to _second()
__second() was post to _first()
>>> test()
Call the function __second() in __first().
Call the function test() in __second().
hello world #print in __second().
None #print in __first().
>>>

如上,第35行輸出后調用__second(),而__second()中又調用了test()并print test(),而后返回__first()中繼續(xù)執(zhí)行print,而這個print語句print的內容是__second()返回的None

它等同于:

復制代碼 代碼如下:

>>> def test():
    return 'hello world'

>>> test=second(True)(test)
test() was post to _second()
>>>
>>> test
<function __second at 0x000000000316D2E8>
>>> test=first(True)(test)
__second() was post to _first()
>>> test
<function __first at 0x0000000003344C18>
>>>


3.多重裝飾器的應用:

比如你是項目經理,你要求每一個代碼塊都必須有參數檢查ArgsType和責任檢查ResponsibilityRegister,這樣就需要兩個裝飾器對此代碼塊進行監(jiān)督。

復制代碼 代碼如下:

#coding=utf-8
import os,sys,re
from collections import OrderedDict

def ArgsType(*argTypes,**kwTypes):
    u'''ArgsType(*argTypes,**kwTypes)
    options=[('opt_UseTypeOfDefaultValue',False)]

    以下為本函數相關的開關,并非類型檢驗相關的關鍵字參數,所有options:
    opt_UseTypeOfDefaultValue=>bool:False,為True時,將對沒有指定類型的帶默
                               認值的參數使用其默認值的類型
    '''
    def _ArgsType(func):
        #確定所有的parameter name
        argNames=func.func_code.co_varnames[:func.func_code.co_argcount]
        #確定所有的default parameter
        defaults=func.func_defaults
        if defaults:
            defaults=dict(zip(argNames[-len(defaults):],defaults))
        else:defaults=None
        #將“參數類型關鍵字參數”中的所有“options關鍵字參數”提出
        options=dict()
        for option,default in [('opt_UseTypeOfDefaultValue',False)]:
            options[option]=kwTypes.pop(option,default)
        #argTypes和kwTypes的總長度應該與argNames一致
        if len(argTypes)+len(kwTypes)>len(argNames):
            raise Exception('Too much types to check %s().'%func.func_name)
        #所有kwTypes中的鍵不能覆蓋在argTypes中已經占用的names
        if not set(argNames[len(argTypes):]).issuperset(
            set(kwTypes.keys())):
            raise Exception('There is some key in kwTypes '+
                'which is not in argNames.')
        #確定所有的參數應該有的types
        types=OrderedDict()
        for name in argNames:types[name]=None
        if len(argTypes):
            for i in range(len(argTypes)):
                name=argNames[i]
                types[name]=argTypes[i]
        else:
            for name,t in kwTypes.items():
                types[name]=t
        if len(kwTypes):
            for name,t in kwTypes.items():
                types[name]=t
        #關于default parameter的type
        if options['opt_UseTypeOfDefaultValue']:
            for k,v in defaults.items():
                #如果default parameter的type沒有另外指定,那么就使用
                #default parameter的default value的type
                if types[k]==None:
                    types[k]=type(v)
        def __ArgsType(*args,**kw):
            #order the args
            Args=OrderedDict()
            #init keys
            for name in argNames:Args[name]=None
            #init default values
            if defaults is not None:
                for k,v in defaults.items():
                    Args[k]=v
            #fill in all args
            for i in range(len(args)):
                Args[argNames[i]]=args[i]
            #fill in all keyword args
            for k,v in kw.items():
                Args[k]=v
            #check if there is some None in the values
            if defaults==None:
                for k in Args:
                    if Args[k]==None:
                        if defaults==None:
                            raise Exception(('%s() needs %r parameter, '+
                                'which was not given')%(func.func_name,k))
                        else:
                           if not defaults.has_key(k):
                                raise Exception(('Parameter %r of %s() is'+
                                    ' not a default parameter')%\
                                    (k,func.func_name))
            #check all types
            for k in Args:
                if not isinstance(Args[k],types[k]):
                    raise TypeError(('Parameter %r of %s() must be '+
                        'a %r object, but you post: %r')%\
                        (k,func.func_name,types[k],Args[k]))
            return func(*args,**kw)
        return __ArgsType
    return _ArgsType

def ResponsibilityRegister(author):
    def _ResponsibilityRegister(func):
        def __ResponsibilityRegister(*args,**kw):
            try:
                return func(*args,**kw)
            except Exception as e:
                print ("Something is wrong, It's %s's responsibility."%\
                       author).center(80,'*')
                raise e
        return __ResponsibilityRegister
    return _ResponsibilityRegister

@ResponsibilityRegister('Kate')
@ArgsType(str,int)
def left(Str,Len=1):
    return Str[:Len]

print 'Good calling:'
print left('hello world',8)
print 'Bad calling:'
print left(3,7)

這里沒有文檔,所以調用者不知道,使用了錯誤的調用,導致出錯,這是Kate的責任。

像上面這種,對代碼有兩種互不相干的檢驗時,就可以使用多重裝飾器。

相關文章

  • Python處理缺失值的8種不同方法實例

    Python處理缺失值的8種不同方法實例

    缺失值是指粗糙數據中由于缺少信息而造成的數據的聚類、分組、刪失或截斷,下面這篇文章主要給大家介紹了關于Python處理缺失值的8種不同方法,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-06-06
  • Python機器學習之隨機梯度下降法的實現

    Python機器學習之隨機梯度下降法的實現

    如果當我們數據量和樣本量非常大時,每一項都要參與到梯度下降,那么它的計算量時非常大的,所以我們需要采用隨機梯度下降法。本文介紹了Python實現隨機梯度下降法的方法,希望對大家有所幫助
    2023-02-02
  • 淺談使用Python內置函數getattr實現分發(fā)模式

    淺談使用Python內置函數getattr實現分發(fā)模式

    這篇文章主要介紹了淺談使用Python內置函數getattr實現分發(fā)模式,小編覺得還是挺不錯的,具有一定借鑒價值,需要的朋友可以參考下
    2018-01-01
  • 利用Python通過獲取剪切板數據實現百度劃詞搜索功能

    利用Python通過獲取剪切板數據實現百度劃詞搜索功能

    大家是不是嫌棄每次打開百度太麻煩?今天教大家利用Python通過獲取剪切板數據實現百度劃詞搜索功能,用程序直接打開網頁,需要的朋友可以參考下
    2021-06-06
  • Python 占位符的使用方法詳解

    Python 占位符的使用方法詳解

    這篇文章主要介紹了Python 占位符的使用方法詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-07-07
  • python實現向ppt文件里插入新幻燈片頁面的方法

    python實現向ppt文件里插入新幻燈片頁面的方法

    這篇文章主要介紹了python實現向ppt文件里插入新幻燈片頁面的方法,涉及Python操作ppt文檔添加幻燈片的相關技巧,非常具有實用價值,需要的朋友可以參考下
    2015-04-04
  • python逆向微信指數爬取實現步驟

    python逆向微信指數爬取實現步驟

    這篇文章主要為大家介紹了python逆向微信指數爬取的實現步驟,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步早日升職加薪
    2022-02-02
  • Python程序中用csv模塊來操作csv文件的基本使用教程

    Python程序中用csv模塊來操作csv文件的基本使用教程

    這篇文章主要介紹了Python程序中用csv模塊來操作csv文件的基本使用教程,csv文件中也是格式化的數據,只不過csv本身沒有XML和JSON那么流行...需要的朋友可以參考下
    2016-03-03
  • Python3中的tuple函數知識點講解

    Python3中的tuple函數知識點講解

    在本篇文章里小編給大家整理了一篇關于Python3中的tuple函數知識點講解內容,有興趣的朋友們可以學習參考下。
    2021-01-01
  • Python學習筆記之解析json的方法分析

    Python學習筆記之解析json的方法分析

    這篇文章主要介紹了Python解析json的方法,結合實例形式分析了常見的Python解析與轉換json格式數據相關操作技巧,需要的朋友可以參考下
    2017-04-04

最新評論