电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術(shù)文章
文章詳情頁

通俗講解python 裝飾器

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

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

其實(shí)在平常寫寫腳本的過程中,這個知識點(diǎn)你可能用到不多

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

一、什么是裝飾器

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

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

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

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

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

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

那么,可以這樣'增強(qiáng)'一下:

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

運(yùn)行結(jié)果:

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

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

當(dāng)運(yùn)行最后的hello()函數(shù)時,調(diào)用過程是這樣的:

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

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

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

二、優(yōu)雅的裝飾器

所以,想讓上述裝飾器變得優(yōu)雅,可以這樣寫:

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

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

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

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]

三、帶參數(shù)的裝飾器

1. 單個參數(shù)

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

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

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. 多個參數(shù)

但是還沒完,這樣雖然簡單,但是隨之而來另一個問題:因?yàn)椴⒉皇撬泻瘮?shù)參數(shù)都是一樣的,當(dāng)其他要使用裝飾器的函數(shù)參數(shù)不止這個一個腫么辦?比如:

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

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

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('張三', '李四')

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

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

3. 自定義參數(shù)

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

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裝飾函數(shù)中的2個return.運(yùn)行下,應(yīng)該會出現(xiàn)3次:

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

4. 內(nèi)置裝飾器@functools.wrap

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

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函數(shù)的元信息

輸出:

wrapper

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

如果我們需要用到元函數(shù)信息,那怎么保留它呢?這時候可以用內(nèi)置裝飾器@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__)

運(yùn)行下:

hello[Finished in 0.1s]

四、類裝飾器

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

但是類作為裝飾器的話,需要依賴一個函數(shù)__call__(),當(dāng)調(diào)用這個類的實(shí)例時,函數(shù)__call__()就會被執(zhí)行。

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

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()

運(yùn)行:

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

跟函數(shù)裝飾器一樣,實(shí)現(xiàn)一樣的功能。

五、裝飾器的嵌套

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

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

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

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()

運(yùn)行

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

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

以上就是通俗講解python 裝飾器的詳細(xì)內(nèi)容,更多關(guān)于python 裝飾器的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Python 編程
相關(guān)文章:
主站蜘蛛池模板: 缠膜机|缠绕包装机|无纺布包装机-济南达伦特机械设备有限公司 | 北京森语科技有限公司-模型制作专家-展览展示-沙盘模型设计制作-多媒体模型软硬件开发-三维地理信息交互沙盘 | 爆破器材运输车|烟花爆竹运输车|1-9类危险品厢式运输车|湖北江南专用特种汽车有限公司 | 线粒体膜电位荧光探针-细胞膜-标记二抗-上海复申生物科技有限公司 | 纯化水设备-纯水设备-超纯水设备-[大鹏水处理]纯水设备一站式服务商-东莞市大鹏水处理科技有限公司 | 礼堂椅厂家|佛山市艺典家具有限公司 | 深圳希玛林顺潮眼科医院(官网)│深圳眼科医院│医保定点│香港希玛林顺潮眼科中心连锁品牌 | 避光流动池-带盖荧光比色皿-生化流动比色皿-宜兴市晶科光学仪器 东莞爱加真空科技有限公司-进口真空镀膜机|真空镀膜设备|Polycold维修厂家 | 搜活动房网—活动房_集装箱活动房_集成房屋_活动房屋 | 数码听觉统合训练系统-儿童感觉-早期言语评估与训练系统-北京鑫泰盛世科技发展有限公司 | 陶瓷砂磨机,盘式砂磨机,棒销式砂磨机-无锡市少宏粉体科技有限公司 | 动物解剖台-成蚊接触筒-标本工具箱-负压实验台-北京哲成科技有限公司 | 苏州教学设备-化工教学设备-环境工程教学模型|同科教仪 | 宿松新闻网 宿松网|宿松在线|宿松门户|安徽宿松(直管县)|宿松新闻综合网站|宿松官方新闻发布 | 铁盒_铁罐_马口铁盒_马口铁罐_铁盒生产厂家-广州博新制罐 | 干培两用箱-细菌恒温培养箱-菲斯福仪器 | 真空粉体取样阀,电动楔式闸阀,电动针型阀-耐苛尔(上海)自动化仪表有限公司 | 顶空进样器-吹扫捕集仪-热脱附仪-二次热解吸仪-北京华盛谱信仪器 | 铝箔袋,铝箔袋厂家,东莞铝箔袋,防静电铝箔袋,防静电屏蔽袋,防静电真空袋,真空袋-东莞铭晋让您的产品与众不同 | H型钢切割机,相贯线切割机,数控钻床,数控平面钻,钢结构设备,槽钢切割机,角钢切割机,翻转机,拼焊矫一体机 | 冰晶石|碱性嫩黄闪蒸干燥机-有机垃圾烘干设备-草酸钙盘式干燥机-常州市宝康干燥 | 防火门|抗爆门|超大门|医疗门|隔声门-上海加汇门业生产厂家 | 润滑油加盟_润滑油厂家_润滑油品牌-深圳市沃丹润滑科技有限公司 琉璃瓦-琉璃瓦厂家-安徽盛阳新型建材科技有限公司 | 郑州外墙清洗_郑州玻璃幕墙清洗_郑州开荒保洁-河南三恒清洗服务有限公司 | 塑胶跑道_学校塑胶跑道_塑胶球场_运动场材料厂家_中国塑胶跑道十大生产厂家_混合型塑胶跑道_透气型塑胶跑道-广东绿晨体育设施有限公司 | 老房子翻新装修,旧房墙面翻新,房屋防水补漏,厨房卫生间改造,室内装潢装修公司 - 一修房屋快修官网 | 青岛球场围网,青岛车间隔离网,青岛机器人围栏,青岛水源地围网,青岛围网,青岛隔离栅-青岛晟腾金属制品有限公司 | loft装修,上海嘉定酒店式公寓装修公司—曼城装饰 | 济南ISO9000认证咨询代理公司,ISO9001认证,CMA实验室认证,ISO/TS16949认证,服务体系认证,资产管理体系认证,SC食品生产许可证- 济南创远企业管理咨询有限公司 郑州电线电缆厂家-防火|低压|低烟无卤电缆-河南明星电缆 | 服务器之家 - 专注于服务器技术及软件下载分享 | 东莞螺杆空压机_永磁变频空压机_节能空压机_空压机工厂批发_深圳螺杆空压机_广州螺杆空压机_东莞空压机_空压机批发_东莞空压机工厂批发_东莞市文颖设备科技有限公司 | 临时厕所租赁_玻璃钢厕所租赁_蹲式|坐式厕所出租-北京慧海通 | 石牌坊价格石牌坊雕刻制作_石雕牌坊牌楼石栏杆厂家_山东嘉祥石雕有限公司 | 清水混凝土修复_混凝土色差修复剂_混凝土色差调整剂_清水混凝土色差修复_河南天工 | 北京京云律师事务所| 档案密集架,移动密集架,手摇式密集架,吉林档案密集架-厂家直销★价格公道★质量保证 | 焦作网 WWW.JZRB.COM| 德国GMN轴承,GMN角接触球轴承,GMN单向轴承,GMN油封,GMN非接触式密封 | 青岛侦探_青岛侦探事务所_青岛劝退小三_青岛调查出轨取证公司_青岛婚外情取证-青岛探真调查事务所 | 杰恒蠕动泵-蠕动泵专业厂家-19年专注蠕动泵 | 压滤机-洗沙泥浆处理-压泥机-山东创新华一环境工程有限公司 |