下面為大家分享一篇python裝飾器-限制函數呼叫次數的方法(10s呼叫一次),具有很好的參考價值,希望對大家有幫助。一起來看看吧
這是部落客最近一家大公司的面試題,寫一個裝飾器,限制函數每10s調用一次。當時是筆試的,只寫了大概的程式碼,回來後溫習了python裝飾器的基礎知識,把程式碼寫完了。決定寫篇部落格記錄下。
裝飾器分為帶參數得裝飾器以及不帶參數得裝飾器。
#不带参数的装饰器 @dec1 @dec2 def func(): ... #这个函数声明等价于 func = dec1(dec2(func)) #带参数的装饰器 @dec(some_args) def func(): ... #这个函数声明等价于 func = dec(some_args)(func)
不帶參數的裝飾器需要注意的一些細節
1. 關於裝飾器函數(decorator)本身
因此一個裝飾器一般對應兩個函數,一個是decorator函數,用來進行一些初始化操作處理,一個是decorated_func用來實現對被裝飾的函數func的額外處理。並且為了保持對func的引用,decorated_func一般作為decorator的內部函數
#def decorator(func): def decorator_func() func() return decorated_func
decorator函數只在函數宣告的時候被呼叫一次
裝飾器實際上是語法糖,在宣告函數之後就會被調用,產生decorated_func,並把func符號的引用替換為decorated_func。之後每次都呼叫func函數,實際呼叫的是decorated_func(這個很重要,裝飾之後,其實每次呼叫的是decorated_func)。
>>> def decorator(func): ... def decorated_func(): ... func(1) ... return decorated_func ... #声明时就被调用 >>> @decorator ... def func(x): ... print x ... decorator being called #使用func()函数实际上使用的是decorated_func函数 >>> func() 1 >>> func.__name__ 'decorated_func'
如果要保證傳回的decorated_func的函數名稱與func的函數名稱相同,應在decorator函數傳回decorated_func之前,加入decorated_func.name = func. name, 另外functools模組提供了wraps裝飾器,可以完成這一動作。
#@wraps(func)的操作相当于 #在return decorated_func之前,执行 #decorated_func.__name__ = func.__name__ #func作为装饰器参数传入, #decorated_func则作为wraps返回的函数的参数传入 >>> def decorator(func): ... @wraps(func) ... def decorated_func(): ... func(1) ... return decorated_func ... #声明时就被调用 >>> @decorator ... def func(x): ... print x ... decorator being called #使用func()函数实际上使用的是decorated_func函数 >>> func() 1 >>> func.__name__ 'func'
decorator函數局部變數的妙用
因為closure的特性(詳見(1)部分閉包部分的詳解),decorator宣告的變數會被decorated_func.func_closure引用,所以呼叫了decorator方法結束之後,decorator方法的局部變數也不會被回收,因此可以用decorator方法的局部變數作為計數器,緩存等等。
值得注意的是,如果要改變變數的值,則變數一定要是可變對象,因此就算是計數器,也應當用列表來實現。且宣告一次函數呼叫一次decorator函數,所以不同函數的計數器之間互不衝突,例如:
##!/usr/bin/env python #filename decorator.py def decorator(func): #注意这里使用可变对象 a = [0] def decorated_func(*args,**keyargs): func(*args, **keyargs) #因为闭包是浅拷贝,如果是不可变对象,每次调用完成后符号都会被清空,导致错误 a[0] += 1 print "%s have bing called %d times" % (func.__name__, a[0]) return decorated_func @decorator def func(x): print x @decorator def theOtherFunc(x): print x
下面我們開始寫程式碼:
#coding=UTF-8 #!/usr/bin/env python #filename decorator.py import time from functools import wraps def decorator(func): "cache for function result, which is immutable with fixed arguments" print "initial cache for %s" % func.__name__ cache = {} @wraps(func) def decorated_func(*args,**kwargs): # 函数的名称作为key key = func.__name__ result = None #判断是否存在缓存 if key in cache.keys(): (result, updateTime) = cache[key] #过期时间固定为10秒 if time.time() -updateTime < 10: print "limit call 10s", key result = updateTime else : print "cache expired !!! can call " result = None else: print "no cache for ", key #如果过期,或则没有缓存调用方法 if result is None: result = func(*args, **kwargs) cache[key] = (result, time.time()) return result return decorated_func @decorator def func(x): print 'call func'
隨便測試了下,基本上沒有問題。
>>> from decorator import func initial cache for func >>> func(1) no cache for func call func >>> func(1) limit call 10s func 1488082913.239092 >>> func(1) cache expired !!! can call call func >>> func(1) limit call 10s func 1488082923.298204 >>> func(1) cache expired !!! can call call func >>> func(1) limit call 10s func 1488082935.165979 >>> func(1) limit call 10s func 1488082935.165979
相關推薦:
########
以上是python裝飾器-限制函數呼叫次數的方法(10s呼叫一次)的詳細內容。更多資訊請關注PHP中文網其他相關文章!