国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Python > 正文

python中的裝飾器詳解

2019-11-25 17:46:13
字體:
來源:轉載
供稿:網友

在了解裝飾器的之前一定要先了解函數作為參數傳遞, 什么是函數內嵌,請參考我之前寫的博客函數簡介

因為在python里面,函數也是對象,也可以作為參數進行傳遞.python裝飾器本質也是一種特殊函數,它接收的參數是函數對象,然后動態地函數參數添加額外的功能,而不用修改原有的函數對象.python裝飾器傳入的參數是函數,返回的值也是函數!
python裝飾器思想有點類似設計模式的裝飾模式, 其意圖是動態地給函數對象添加額外的功能.比如像增加日志打印的功能,有點面向切面編程(AOP)的感覺.
裝飾器語法

以@開頭,接著后面跟著的是裝飾器的名字和可選的參數.裝飾器語法是一種語法糖.
格式如下

復制代碼 代碼如下:

@decomaker(deco_args)
    def foo(func_opt_args)

可以組合,等價于foo = g(f(foo))
復制代碼 代碼如下:

@g
@f
def foo():
    statement

簡單裝飾器

實例

復制代碼 代碼如下:

#!/usr/bin/python
def  deco(func):
    print 'start'
    func()
    print 'end'
    return func

@deco
def foo():
    print 'In foo'

foo()
foo()


輸出
復制代碼 代碼如下:

start
In foo
end
In foo
In foo

帶內嵌函數裝飾器

內嵌函數保證每次新函數都被調用.而且被裝飾的函數可以帶有參數.
實例

復制代碼 代碼如下:

def  deco(func):
    def _deco(x):    #該函數為內嵌函數
        print 'start'
        func(x)
        print 'end'
    return _deco

@deco
def foo(x):
    print 'In foo, get value is: %d' % x

foo(123456)


輸出:
復制代碼 代碼如下:

start
In foo, get value is: 123456
end

帶參數的裝飾器

需要自己返回以函數作為參數的裝飾器。換句話說,decomaker()用 deco_args 做了些事并返回函數對象,而該函數對象正是以 foo 作為其參數的裝飾器。簡單的說來:foo=decomaker(deco_args)(foo)

實例

復制代碼 代碼如下:

def deco(arg):
    def wrapper1(func):
        def _deco(x):
            print "get type is: ", arg
            func(x)
        return _deco

    def wrapper2(func):
        def _deco(x):
            func(x)
            print "get type is: ", arg
        return _deco

    if arg == 'type1':
        return wrapper1
    else:
        return wrapper2

@deco("type2")
def foo(x):
    print 'In foo: ', x

foo(123)


輸出
復制代碼 代碼如下:

In foo:  123
get type is:  type2

總結

裝飾器本質是高階的函數,可以裝飾其他函數,增加被裝飾函數的功能,但不能覆蓋或改變被裝飾函數原有的行為.對于被裝飾的函數來說,裝飾器是透明的.裝飾器傳入參數為函數,返回的函數是被裝飾的函數.最后我們來實現給一個函數添加打印日志的功能,而不用改變這個函數.

復制代碼 代碼如下:

#!/usr/bin/python
#coding=utf-8
import functools

def log(prefix, suffix):
    def deco(func):
        @functools.wraps(func)
        def wrapper(*args, **kargs):
            print '%s log start' % prefix
            print('get a is: %s' % args[0])
            print('get b is: %s' % args[1])
            print('get c is: %s' % args[2])
            print('get d is: %s' % kargs['d'])
            print('get d is: %s' % kargs['f'])
            func(*args, **kargs)
            print '%s log end' % suffix
        return wrapper
    return deco

@log('logstart', 'logend')
def test(a, b, c, d, f):
    print 'call func name is: %s' % test.__name__

test(1, 2, 3, d = 'dddd', f = 'ffff')


輸出:
復制代碼 代碼如下:

logstart log start
get a is: 1
get b is: 2
get c is: 3
get d is: dddd
get d is: ffff
call func name is: test
logend log end

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 白水县| 河曲县| 定南县| 南昌市| 原平市| 澜沧| 大荔县| 观塘区| 武穴市| 武胜县| 定襄县| 怀柔区| 读书| 邹城市| 彭山县| 平邑县| 武安市| 万州区| 海安县| 雅江县| 唐山市| 鸡西市| 华坪县| 望谟县| 闸北区| 花垣县| 攀枝花市| 喀喇沁旗| 寿宁县| 德清县| 正蓝旗| 泾源县| 本溪市| 库尔勒市| 罗甸县| 遵义县| 西乡县| 大港区| 汉寿县| 西畴县| 陆河县|