亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

通俗講解python 裝飾器

瀏覽:2日期:2022-07-12 09:47:56

裝飾器其實一直是我的一個'老大難'。這個知識點就放在那,但是拖延癥。。。

其實在平常寫寫腳本的過程中,這個知識點你可能用到不多

但在面試的時候,這可是一個高頻問題。

一、什么是裝飾器

所謂的裝飾器,其實就是通過裝飾器函數,來修改原函數的一些功能,使得原函數不需要修改。

這一句話理解起來可能沒那么輕松,那先來看一個'傻瓜'函數。

放心,絕對不是'Hello World'!

def hello(): print('你好,裝飾器')

腫么樣,木騙你吧? 哈哈,這個函數不用運行相信大家都知道輸出結果:'你好,裝飾器'。

那如果我想讓hello()函數再實現個其他功能,比如多打印一句話。

那么,可以這樣'增強'一下:

def my_decorator(func): def wrapper(): print('這是裝飾后具有的新輸出') func() return wrapperdef hello(): print('你好,裝飾器')hello = my_decorator(hello)hello()

運行結果:

這是裝飾后具有的新輸出你好,裝飾器[Finished in 0.1s]

很顯然,這個'增強'沒啥作用,但是可以幫助理解裝飾器。

當運行最后的hello()函數時,調用過程是這樣的:

hello = my_decorator(hello)中,變量hello指向的是my_decorator() my_decorator(func)中傳參是hello,返回的wrapper,因此又會調用到原函數hello() 于是乎,先打印出了wrapper()函數里的,然后才打印出hello()函數里的

那上述代碼里的my_decorator()就是一個裝飾器。它改變了hello()的行為,但是并沒有去真正的改變hello()函數的內部實現。

但是,python一直以'優雅'被人追捧,而上述的代碼顯然不夠優雅。

二、優雅的裝飾器

所以,想讓上述裝飾器變得優雅,可以這樣寫:

def my_decorator(func): def wrapper(): print('這是裝飾后具有的新輸出') func() return wrapper@my_decoratordef hello(): print('你好,裝飾器')hello()

這里的@my_decorator就相當于舊代碼的hello = my_decorator(hello),@符號稱為語法糖。

那如果還有其他函數也需要加上類似的裝飾,直接在函數的上方加上@my_decorator就可以,大大提高函數的重復利用與可讀性。

def my_decorator(func): def wrapper(): print('這是裝飾后具有的新輸出') func() return wrapper@my_decoratordef hello(): print('你好,裝飾器')@my_decoratordef hello2(): print('你好,裝飾器2')hello2()

輸出:

這是裝飾后具有的新輸出你好,裝飾器2[Finished in 0.1s]

三、帶參數的裝飾器

1. 單個參數

上面的只是一個非常簡單的裝飾器,但是實際場景中,很多函數都是要帶有參數的,比如hello(people_name)。

其實也很簡單,要什么我們就給什么唄,直接在對應裝飾器的wrapper()上,加上對應的參數:

def my_decorator(func): def wrapper(people_name): print('這是裝飾后具有的新輸出') func(people_name) return wrapper@my_decoratordef hello(people_name): print('你好,{}'.format(people_name))hello('張三')

輸出:

這是裝飾后具有的新輸出你好,張三[Finished in 0.1s]

2. 多個參數

但是還沒完,這樣雖然簡單,但是隨之而來另一個問題:因為并不是所有函數參數都是一樣的,當其他要使用裝飾器的函數參數不止這個一個腫么辦?比如:

@my_decoratordef hello3(speaker, listener): print('{}對{}說你好!'.format(speaker, listener))

沒關系,在python里,*args和**kwargs表示接受任意數量和類型的參數,所以我們可以這樣寫裝飾器里的wrapper()函數:

def my_decorator(func): def wrapper(*args, **kwargs): print('這是裝飾后具有的新輸出') func(*args, **kwargs) return wrapper@my_decoratordef hello(people_name): print('你好,{}'.format(people_name))@my_decoratordef hello3(speaker, listener): print('{}對{}說你好!'.format(speaker, listener))hello('老王')print('------------------------')hello3('張三', '李四')

同時運行下hello('老王'),和hello3('張三', '李四'),看結果:

這是裝飾后具有的新輸出你好,老王------------------------這是裝飾后具有的新輸出張三對李四說你好![Finished in 0.1s]

3. 自定義參數

上面2種,裝飾器都是接收外來的參數,其實裝飾器還可以接收自己的參數。比如,我加個參數來控制下裝飾器中打印信息的次數:

def count(num): def my_decorator(func): def wrapper(*args, **kwargs): for i in range(num):print('這是裝飾后具有的新輸出')func(*args, **kwargs) return wrapper return my_decorator@count(3)def hello(people_name): print('你好,{}'.format(people_name))hello('老王')

注意,這里count裝飾函數中的2個return.運行下,應該會出現3次:

這是裝飾后具有的新輸出你好,老王這是裝飾后具有的新輸出你好,老王這是裝飾后具有的新輸出你好,老王[Finished in 0.1s]

4. 內置裝飾器@functools.wrap

現在多做一步探索,我們來打印下下面例子中的hello()函數的元信息:

def my_decorator(func): def wrapper(*args, **kwargs): print('這是裝飾后具有的新輸出') func(*args, **kwargs) return wrapper@my_decoratordef hello(people_name): print('你好,{}'.format(people_name))print(hello.__name__) #看下hello函數的元信息

輸出:

wrapper

這說明了,它不再是以前的那個 hello() 函數,而是被 wrapper() 函數取代了。

如果我們需要用到元函數信息,那怎么保留它呢?這時候可以用內置裝飾器@functools.wrap。

import functoolsdef my_decorator(func): @functools.wraps(func) def wrapper(*args, **kwargs): print('這是裝飾后具有的新輸出') func(*args, **kwargs) return wrapper@my_decoratordef hello(people_name): print('你好,{}'.format(people_name))print(hello.__name__)

運行下:

hello[Finished in 0.1s]

四、類裝飾器

裝飾器除了是函數之外,也可以是類。

但是類作為裝飾器的話,需要依賴一個函數__call__(),當調用這個類的實例時,函數__call__()就會被執行。

來改造下之前的例子,把函數裝飾器改成類裝飾器:

class MyDecorator(): def __init__(self, func): self.func = func def __call__(self, *args, **kwargs): print('這是裝飾后具有的新輸出') return self.func(*args, **kwargs)# def my_decorator(func):# def wrapper():# print('這是裝飾后具有的新輸出')# func()# return wrapper@MyDecoratordef hello(): print('你好,裝飾器')hello()

運行:

這是裝飾后具有的新輸出你好,裝飾器[Finished in 0.1s]

跟函數裝飾器一樣,實現一樣的功能。

五、裝飾器的嵌套

既然裝飾器可以增強函數的功能,那如果有多個裝飾器,我都想要怎么辦?其實,只要把需要用的裝飾器都加上去就好了:

@decorator1@decorator2@decorator3def hello(): ...

但是要注意這里的執行順序,會從上到下去執行,可以來看下:

def my_decorator(func): def wrapper(): print('這是裝飾后具有的新輸出') func() return wrapperdef my_decorator2(func): def wrapper(): print('這是裝飾后具有的新輸出2') func() return wrapperdef my_decorator3(func): def wrapper(): print('這是裝飾后具有的新輸出3') func() return wrapper@my_decorator@my_decorator2@my_decorator3def hello(): print('你好,裝飾器')hello()

運行

這是裝飾后具有的新輸出這是裝飾后具有的新輸出2這是裝飾后具有的新輸出3你好,裝飾器[Finished in 0.1s]

好記性不如爛筆頭,寫一下理解一下會好很多。

以上就是通俗講解python 裝飾器的詳細內容,更多關于python 裝飾器的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
主站蜘蛛池模板: 国产限制路线1线路2线路3 | 国产成人免费无庶挡视频 | 欧美黑人成人www在线观看 | 69国产成人精品午夜福中文 | 九九视频免费观看 | 亚洲国产一区在线 | 国产拍| 蝌蚪蚪窝视频在线视频手机 | 青青草国产精品人人爱99 | 福利视频在线观看免费版 | 亚洲狠狠成人综合网 | 精品午夜久久影视 | 一区在线视频 | 成人夜视频 | 免费观看全黄做爰 | 爱爱爱爱的视频在线观看 | 久久久久久久久久免免费精品 | 中国人免费的片 | 婷婷亚洲综合一区二区 | 午夜亚洲 | 91久久精一区二区三区大全 | 国产高清不卡一区二区三区 | 成人精品视频在线观看播放 | 香蕉视频在线观看黄 | 国产精品日韩欧美 | 韩日在线播放 | 欧美ol丝袜高跟秘书在线播放 | 久久厕所精品国产精品亚洲 | 黄网站免费在线观看 | 日韩a级毛片免费观看 | 午夜精品久久久久久91 | 欧美一级级a在线观看 | 日日夜操 | 国产乱人免费视频 | 亚洲色图日韩 | 91爱啪啪| 那个网站可以看毛片 | 午夜在线观看视频免费 成人 | 国产成人欧美一区二区三区的 | 亚洲色图吧 | 国产精品资源手机在线播放 |