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

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

淺談python中的@以及@在tensorflow中的作用說明

瀏覽:5日期:2022-06-26 11:52:04

雖然用python用了很久了,但是主要還是寫一些模型或者算子,對于python中的高級特性用的不多,但是時常閱讀大牛的代碼或者框架源碼,其中python特性應(yīng)用的非常流暢,所以今天決定與python中的裝飾器@,做個了斷??!

Python中的@:

援引廖雪峰老師對裝飾器的解釋以及一些自己對裝飾器的理解:

python中在代碼運行期間動態(tài)增加功能的方式,稱之為“裝飾器”(Decorator)。@是裝飾器的語法。裝飾器是在函數(shù)調(diào)用之上的修飾,這些修飾僅是當(dāng)聲明一個函數(shù)或者方法的時候,才會應(yīng)用的額外調(diào)用。 我們可以用裝飾器來增加計時邏輯來檢測性能,或者引入日志等等。

函數(shù)也是一個對象,而且函數(shù)對象可以被賦值給變量,所以,通過變量也能調(diào)用該函數(shù)。

>>> def now():... print(’2015-3-25’)...>>> f = now>>> f()2015-3-25

函數(shù)對象有一個__name__屬性,可以拿到函數(shù)的名字:

>>> now.__name__’now’>>> f.__name__’now’

現(xiàn)在,假設(shè)我們要增強now()函數(shù)的功能,比如,在函數(shù)調(diào)用前后自動打印日志,但又不希望修改now()函數(shù)的定義。本質(zhì)上,decorator就是一個返回函數(shù)的高階函數(shù)。所以,我們要定義一個能打印日志的decorator,可以定義如下:

def log(func): def wrapper(*args, **kw): print(’call %s():’ % func.__name__) return func(*args, **kw) return wrapper

觀察上面的log,因為它是一個decorator,所以接受一個函數(shù)作為參數(shù),并返回一個函數(shù)。我們要借助Python的@語法,把decorator置于函數(shù)的定義處:

@logdef now(): print(’2015-3-25’)

調(diào)用now()函數(shù),不僅會運行now()函數(shù)本身,還會在運行now()函數(shù)前打印一行日志:

>>> now()call now():2015-3-25

把@log放到now()函數(shù)的定義處,相當(dāng)于執(zhí)行了語句:

now = log(now)

由于log()是一個decorator,返回一個函數(shù),所以,原來的now()函數(shù)仍然存在,只是現(xiàn)在同名的now變量指向了新的函數(shù),于是調(diào)用now()將執(zhí)行新函數(shù),即在log()函數(shù)中返回的wrapper()函數(shù)。

wrapper()函數(shù)的參數(shù)定義是(*args, **kw),因此,wrapper()函數(shù)可以接受任意參數(shù)的調(diào)用。在wrapper()函數(shù)內(nèi),首先打印日志,再緊接著調(diào)用原始函數(shù)。

如果decorator本身需要傳入?yún)?shù),那就需要編寫一個返回decorator的高階函數(shù),寫出來會更復(fù)雜。比如,要自定義log的文本:

def log(text): def decorator(func): def wrapper(*args, **kw): print(’%s %s():’ % (text, func.__name__)) return func(*args, **kw) return wrapper return decorator

這個3層嵌套的decorator用法如下:

@log(’execute’)def now(): print(’2015-3-25’)

執(zhí)行結(jié)果如下:

>>> now()execute now():2015-3-25

和兩層嵌套的decorator相比,3層嵌套的效果是這樣的:

>>> now = log(’execute’)(now)

我們來剖析上面的語句,首先執(zhí)行l(wèi)og(’execute’),返回的是decorator函數(shù),再調(diào)用返回的函數(shù),參數(shù)是now函數(shù),返回值最終是wrapper函數(shù)。

以上兩種decorator的定義都沒有問題,但還差最后一步。因為我們講了函數(shù)也是對象,它有__name__等屬性,但你去看經(jīng)過decorator裝飾之后的函數(shù),它們的__name__已經(jīng)從原來的’now’變成了’wrapper’:

>>> now.__name__’wrapper’

因為返回的那個wrapper()函數(shù)名字就是’wrapper’,所以,需要把原始函數(shù)的__name__等屬性復(fù)制到wrapper()函數(shù)中,否則,有些依賴函數(shù)簽名的代碼執(zhí)行就會出錯。

不需要編寫wrapper.__name__ = func.__name__這樣的代碼,Python內(nèi)置的functools.wraps就是干這個事的,所以,一個完整的decorator的寫法如下:

import functools def log(func): @functools.wraps(func) def wrapper(*args, **kw): print(’call %s():’ % func.__name__) return func(*args, **kw) return wrapper

或者針對帶參數(shù)的decorator:

import functools def log(text): def decorator(func): @functools.wraps(func) def wrapper(*args, **kw): print(’%s %s():’ % (text, func.__name__)) return func(*args, **kw) return wrapper return decorator

import functools是導(dǎo)入functools模塊。模塊的概念稍候講解。現(xiàn)在,只需記住在定義wrapper()的前面加上@functools.wraps(func)即可。

python中常見的@:

@property :對于類的方法,裝飾器一樣起作用,Python內(nèi)置的@property裝飾器就是負(fù)責(zé)把一個方法變成屬性調(diào)用的.廣泛應(yīng)用在類的定義中,可以讓調(diào)用者寫出簡短的代碼,同時保證對參數(shù)進行必要的檢查,這樣,程序運行時就減少了出錯的可能性。

class Student(object): @property def score(self): return self._score @score.setter def score(self, value): if not isinstance(value, int): raise ValueError(’score must be an integer!’) if value < 0 or value > 100: raise ValueError(’score must between 0 ~ 100!’) self._score = value

@property的實現(xiàn)比較復(fù)雜,我們先考察如何使用。把一個getter方法變成屬性,只需要加上@property就可以了,此時,@property本身又創(chuàng)建了另一個裝飾器@score.setter,負(fù)責(zé)把一個setter方法變成屬性賦值,于是,我們就擁有一個可控的屬性操作:

>>> s = Student()>>> s.score = 60 # OK,實際轉(zhuǎn)化為s.set_score(60)>>> s.score # OK,實際轉(zhuǎn)化為s.get_score()60>>> s.score = 9999Traceback (most recent call last): ...ValueError: score must between 0 ~ 100!

注意到這個神奇的@property,我們在對實例屬性操作的時候,就知道該屬性很可能不是直接暴露的,而是通過getter和setter方法來實現(xiàn)的。

@staticmethod,@classmethod:@staticmethod返回的是一個staticmethod類對象,而@classmethod返回的是一個classmethod類對象。他們都是調(diào)用的是各自的__init__()構(gòu)造函數(shù)。

當(dāng)然應(yīng)用裝飾器不當(dāng)也會帶來一些問題:1、位置錯誤的代碼

​ 讓我們直接看示例代碼。

def html_tags(tag_name): print ’begin outer function.’ def wrapper_(func): print 'begin of inner wrapper function.' def wrapper(*args, **kwargs): content = func(*args, **kwargs) print '<{tag}>{content}</{tag}>'.format(tag=tag_name, content=content) print ’end of inner wrapper function.’ return wrapper print ’end of outer function’ return wrapper_ @html_tags(’b’)def hello(name=’Toby’): return ’Hello {}!’.format(name) hello()hello()

在裝飾器中我在各個可能的位置都加上了print語句,用于記錄被調(diào)用的情況。你知道他們最后打印出來的順序嗎?如果你心里沒底,那么最好不要在裝飾器函數(shù)之外添加邏輯功能,否則這個裝飾器就不受你控制了。以下是輸出結(jié)果:

begin outer function.end of outer functionbegin of inner wrapper function.end of inner wrapper function.<b>Hello Toby!</b><b>Hello Toby!</b>2、錯誤的函數(shù)簽名和文檔

裝飾器裝飾過的函數(shù)看上去名字沒變,其實已經(jīng)變了。

def logging(func): def wrapper(*args, **kwargs): '''print log before a function.''' print '[DEBUG] {}: enter {}()'.format(datetime.now(), func.__name__) return func(*args, **kwargs) return wrapper@loggingdef say(something): '''say something''' print 'say {}!'.format(something) print say.__name__ # wrapper

為什么會這樣呢?想想裝飾器的語法@代替的東西就明白了。@等同于這樣的寫法。

say = logging(say)

logging其實返回的函數(shù)名字剛好是wrapper,那么上面的這個語句剛好就是把這個結(jié)果賦值給say,say的__name__自然也就是wrapper了,不僅僅是name,其他屬性也都是來自wrapper,比如doc,source等等。

使用標(biāo)準(zhǔn)庫里的functools.wraps,可以基本解決這個問題。

from functools import wraps def logging(func): @wraps(func) def wrapper(*args, **kwargs): '''print log before a function.''' print '[DEBUG] {}: enter {}()'.format(datetime.now(), func.__name__) return func(*args, **kwargs) return wrapper @loggingdef say(something): '''say something''' print 'say {}!'.format(something) print say.__name__ # sayprint say.__doc__ # say something

看上去不錯!主要問題解決了,但其實還不太完美。因為函數(shù)的簽名和源碼還是拿不到的。

import inspectprint inspect.getargspec(say) # failedprint inspect.getsource(say) # failed

如果要徹底解決這個問題可以借用第三方包,比如wrapt。

3、不能裝飾@staticmethod或者 @classmethod

當(dāng)你想把裝飾器用在一個靜態(tài)方法或者類方法時,不好意思,報錯了。

class Car(object): def __init__(self, model): self.model = model @logging # 裝飾實例方法,OK def run(self): print '{} is running!'.format(self.model) @logging # 裝飾靜態(tài)方法,F(xiàn)ailed @staticmethod def check_model_for(obj): if isinstance(obj, Car): print 'The model of your car is {}'.format(obj.model) else: print '{} is not a car!'.format(obj) '''Traceback (most recent call last):... File 'example_4.py', line 10, in logging @wraps(func) File 'C:Python27libfunctools.py', line 33, in update_wrapper setattr(wrapper, attr, getattr(wrapped, attr))AttributeError: ’staticmethod’ object has no attribute ’__module__’'''

前面已經(jīng)解釋了@staticmethod這個裝飾器,其實它返回的并不是一個callable對象,而是一個staticmethod對象,那么它是不符合裝飾器要求的(比如傳入一個callable對象),你自然不能在它之上再加別的裝飾器。要解決這個問題很簡單,只要把你的裝飾器放在@staticmethod之前就好了,因為你的裝飾器返回的還是一個正常的函數(shù),然后再加上一個@staticmethod是不會出問題的。

class Car(object): def __init__(self, model): self.model = model @staticmethod @logging # 在@staticmethod之前裝飾,OK def check_model_for(obj): pass如何優(yōu)化你的裝飾器:

嵌套的裝飾函數(shù)不太直觀,我們可以使用第三方包類改進這樣的情況,讓裝飾器函數(shù)可讀性更好。

decorator.py

decorator.py是一個非常簡單的裝飾器加強包。你可以很直觀的先定義包裝函數(shù)wrapper(),再使用decorate(func, wrapper)方法就可以完成一個裝飾器。

from decorator import decorate def wrapper(func, *args, **kwargs): '''print log before a function.''' print '[DEBUG] {}: enter {}()'.format(datetime.now(), func.__name__) return func(*args, **kwargs) def logging(func): return decorate(func, wrapper) # 用wrapper裝飾func

你也可以使用它自帶的@decorator裝飾器來完成你的裝飾器。

from decorator import decorator @decoratordef logging(func, *args, **kwargs): print '[DEBUG] {}: enter {}()'.format(datetime.now(), func.__name__) return func(*args, **kwargs)

decorator.py實現(xiàn)的裝飾器能完整保留原函數(shù)的name,doc和args,唯一有問題的就是inspect.getsource(func)返回的還是裝飾器的源代碼,你需要改成inspect.getsource(func.__wrapped__)。

wrapt

wrapt是一個功能非常完善的包,用于實現(xiàn)各種你想到或者你沒想到的裝飾器。使用wrapt實現(xiàn)的裝飾器你不需要擔(dān)心之前inspect中遇到的所有問題,因為它都幫你處理了,甚至inspect.getsource(func)也準(zhǔn)確無誤。

import wrapt# without argument in decorator@wrapt.decoratordef logging(wrapped, instance, args, kwargs): # instance is must print '[DEBUG]: enter {}()'.format(wrapped.__name__) return wrapped(*args, **kwargs) @loggingdef say(something): pass

使用wrapt你只需要定義一個裝飾器函數(shù),但是函數(shù)簽名是固定的,必須是(wrapped, instance, args, kwargs),注意第二個參數(shù)instance是必須的,就算你不用它。當(dāng)裝飾器裝飾在不同位置時它將得到不同的值,比如裝飾在類實例方法時你可以拿到這個類實例。根據(jù)instance的值你能夠更加靈活的調(diào)整你的裝飾器。另外,args和kwargs也是固定的,注意前面沒有星號。在裝飾器內(nèi)部調(diào)用原函數(shù)時才帶星號。

如果你需要使用wrapt寫一個帶參數(shù)的裝飾器,可以這樣寫。

def logging(level): @wrapt.decorator def wrapper(wrapped, instance, args, kwargs): print '[{}]: enter {}()'.format(level, wrapped.__name__) return wrapped(*args, **kwargs) return wrapper @logging(level='INFO')def do(work): passTensorflow中的@:

tensorflow就巧妙應(yīng)用的python的裝飾器,提高了代碼的動態(tài)性,也使代碼變得精簡。

@tf_export 的作用是:Provides ways to export symbols to the TensorFlow API.

@tf_contextlib的作用是:A tf_decorator-aware wrapper for `contextlib.contextmanager`.

還有@tf_inspect、@tf_should_use等。

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持好吧啦網(wǎng)。如有錯誤或未考慮完全的地方,望不吝賜教。

標(biāo)簽: Python 編程
相關(guān)文章:
主站蜘蛛池模板: 泉州陶瓷pc砖_园林景观砖厂家_石英砖地铺石价格 _福建暴风石英砖 | 找果网 | 苹果手机找回方法,苹果iPhone手机丢了找回,认准找果网! | 存包柜厂家_电子存包柜_超市存包柜_超市电子存包柜_自动存包柜-洛阳中星 | 过跨车_过跨电瓶车_过跨转运车_横移电动平车_厂区转运车_无轨转运车 | 依维柯自动挡房车,自行式国产改装房车,小型房车价格,中国十大房车品牌_南京拓锐斯特房车 - 南京拓锐斯特房车 | 全钢实验台,实验室工作台厂家-无锡市辰之航装饰材料有限公司 | 北钻固控设备|石油钻采设备-石油固控设备厂家| 净化车间_洁净厂房_净化公司_净化厂房_无尘室工程_洁净工程装修|改造|施工-深圳净化公司 | 新能源汽车电池软连接,铜铝复合膜柔性连接,电力母排-容发智能科技(无锡)有限公司 | 考勤系统_考勤管理系统_网络考勤软件_政企|集团|工厂复杂考勤工时统计排班管理系统_天时考勤 | 成人纸尿裤,成人尿不湿,成人护理垫-山东康舜日用品有限公司 | 土壤墒情监测站_土壤墒情监测仪_土壤墒情监测系统_管式土壤墒情站-山东风途物联网 | 校车_校车价格_19座幼儿园校车_幼儿园校车_大鼻子校车 | 数显水浴恒温振荡器-分液漏斗萃取振荡器-常州市凯航仪器有限公司 | 小学教案模板_中学教师优秀教案_高中教学设计模板_教育巴巴 | 金属回收_废铜废铁回收_边角料回收_废不锈钢回收_废旧电缆线回收-广东益夫金属回收公司 | 小型高低温循环试验箱-可程式高低温湿热交变试验箱-东莞市拓德环境测试设备有限公司 | 不锈钢散热器,冷却翅片管散热器厂家-无锡市烨晟化工装备科技有限公司 | 非小号行情 - 专业的区块链、数字藏品行情APP、金色财经官网 | 【甲方装饰】合肥工装公司-合肥装修设计公司,专业从事安徽办公室、店面、售楼部、餐饮店、厂房装修设计服务 | 广东燎了网络科技有限公司官网-网站建设-珠海网络推广-高端营销型外贸网站建设-珠海专业h5建站公司「了了网」 | 合肥角钢_合肥槽钢_安徽镀锌管厂家-昆瑟商贸有限公司 | 招商帮-一站式网络营销服务|互联网整合营销|网络推广代运营|信息流推广|招商帮企业招商好帮手|搜索营销推广|短视视频营销推广 | 金属清洗剂,防锈油,切削液,磨削液-青岛朗力防锈材料有限公司 | MVR蒸发器厂家-多效蒸发器-工业废水蒸发器厂家-康景辉集团官网 | 丹佛斯变频器-Danfoss战略代理经销商-上海津信变频器有限公司 | 珠海网站建设_响应网站建设_珠海建站公司_珠海网站设计与制作_珠海网讯互联 | 电动葫芦|环链电动葫芦-北京凌鹰名优起重葫芦| 对照品_中药对照品_标准品_对照药材_「格利普」高纯中药标准品厂家-成都格利普生物科技有限公司 澳门精准正版免费大全,2025新澳门全年免费,新澳天天开奖免费资料大全最新,新澳2025今晚开奖资料,新澳马今天最快最新图库 | 温泉机设备|温泉小镇规划设计|碳酸泉设备 - 大连连邦温泉科技 | 亚克力制品定制,上海嘉定有机玻璃加工制作生产厂家—官网 | 钢制暖气片散热器_天津钢制暖气片_卡麦罗散热器厂家 | 右手官网|右手工业设计|外观设计公司|工业设计公司|产品创新设计|医疗产品结构设计|EMC产品结构设计 | 谷歌关键词优化-外贸网站优化-Google SEO小语种推广-思亿欧外贸快车 | 锂电池砂磨机|石墨烯砂磨机|碳纳米管砂磨机-常州市奥能达机械设备有限公司 | 蓝莓施肥机,智能施肥机,自动施肥机,水肥一体化项目,水肥一体机厂家,小型施肥机,圣大节水,滴灌施工方案,山东圣大节水科技有限公司官网17864474793 | 阁楼货架_阁楼平台_仓库仓储设备_重型货架_广州金铁牛货架厂 | 蓄电池回收,ups电池后备电源回收,铅酸蓄电池回收,机房电源回收-广州益夫铅酸电池回收公司 | 低气压试验箱_高低温低气压试验箱_低气压实验箱 |林频试验设备品牌 | 鑫达滑石-辽宁鑫达滑石集团 | 四川成人高考_四川成考报名网 |