您好,登錄后才能下訂單哦!
這篇文章給大家分享的是有關(guān)python中自帶緩存lru_cache怎么用的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過來看看吧。
以下是lru_cache方法的實(shí)現(xiàn),我們看出可供我們傳入的參數(shù)有2個(gè)maxsize和typed,如果不傳則maxsize的默認(rèn)值為128,typed的默認(rèn)值為False。其中maxsize參數(shù)表示是的被裝飾的方法最大可緩存結(jié)果數(shù)量, 如果是默認(rèn)值128則表示被裝飾方法最多可緩存128個(gè)返回結(jié)果,如果maxsize傳入為None則表示可以緩存無限個(gè)結(jié)果,你可能會(huì)疑惑被裝飾方法的n個(gè)結(jié)果是怎么來的,打個(gè)比方被裝飾的方法為def add(a, b):當(dāng)函數(shù)被lru_cache裝飾時(shí),我們調(diào)用add(1, 2)和add(3, 4)將會(huì)緩存不同的結(jié)果。如果 typed 設(shè)置為true,不同類型的函數(shù)參數(shù)將被分別緩存。例如, f(3) 和 f(3.0) 將被視為不同而分別緩存。
def lru_cache(maxsize=128, typed=False): if isinstance(maxsize, int): if maxsize < 0: maxsize = 0 elif maxsize is not None: raise TypeError('Expected maxsize to be an integer or None') def decorating_function(user_function): wrapper = _lru_cache_wrapper(user_function, maxsize, typed, _CacheInfo) return update_wrapper(wrapper, user_function) return decorating_function
在我們編寫接口時(shí)可能需要緩存一些變動(dòng)不大的數(shù)據(jù)如配置信息,我們可能編寫如下接口:
@api.route("/user/info", methods=["GET"]) @functools.lru_cache() @login_require def get_userinfo_list(): userinfos = UserInfo.query.all() userinfo_list = [user.to_dict() for user in userinfos] return jsonify(userinfo_list)
我們緩存了從數(shù)據(jù)庫查詢的用戶信息,下次再調(diào)用這個(gè)接口時(shí)將直接返回用戶信息列表而不需要重新執(zhí)行一遍數(shù)據(jù)庫查詢邏輯,可以有效較少IO次數(shù),加快接口反應(yīng)速度。
還是以上面的例子,如果發(fā)生用戶的刪除或者新增時(shí),我們再請求用戶接口時(shí)仍然返回的是緩存中的數(shù)據(jù),這樣返回的信息就和我們數(shù)據(jù)庫中的數(shù)據(jù)就會(huì)存在差異,所以當(dāng)發(fā)生用戶新增或者刪除時(shí),我們需要清除原先的緩存,然后再請求用戶接口時(shí)可以重新加載緩存。
@api.route("/user/info", methods=["POST"]) @functools.lru_cache() @login_require def add_user(): user = UserInfo(name="李四") db.session.add(user) db.session.commit() # 清除get_userinfo_list中的緩存 get_userinfo_list = current_app.view_functions["api.get_machine_list"] cache_info = get_userinfo_list.cache_info() # cache_info 具名元組,包含命中次數(shù) hits,未命中次數(shù) misses ,最大緩存數(shù)量 maxsize 和 當(dāng)前緩存大小 currsize # 如果緩存數(shù)量大于0則清除緩存 if cache_info[3] > 0: get_userinfo_list.cache_clear() return jsonify("新增用戶成功")
在上面這個(gè)用法中我們,如果我們把lru_cache裝飾器和login_require裝飾器調(diào)換位置時(shí),上述的寫法將會(huì)報(bào)錯(cuò),這是因?yàn)閘ogin_require裝飾器中用了functiontools.wrap模塊進(jìn)行裝飾導(dǎo)致的,具原因我們在下節(jié)解釋, 如果想不報(bào)錯(cuò)得修改成如下寫法。
@api.route("/user/info", methods=["POST"]) @login_require @functools.lru_cache() def add_user(): user = UserInfo(name="李四") db.session.add(user) db.session.commit() # 清除get_userinfo_list中的緩存 get_userinfo_list = current_app.view_functions["api.get_machine_list"] cache_info = get_userinfo_list.__wrapped__.cache_info() # cache_info 具名元組,包含命中次數(shù) hits,未命中次數(shù) misses ,最大緩存數(shù)量 maxsize 和 當(dāng)前緩存大小 currsize # 如果緩存數(shù)量大于0則清除緩存 if cache_info[3] > 0: get_userinfo_list.__wrapped__.cache_clear() return jsonify("新增用戶成功")
在上節(jié)我們看到,因?yàn)锧login_require和@functools.lru_cache()裝飾器的順序不同, 就導(dǎo)致了程序是否報(bào)錯(cuò), 其中主要涉及到兩點(diǎn):
login_require裝飾器中是否用了@functiontools.wrap()裝飾器
@login_require和@functools.lru_cache()裝飾器的執(zhí)行順序問題
當(dāng)我們了解完這兩點(diǎn)后就可以理解上述寫法了。
這里從其他地方盜了一段代碼來解釋一下,如下:
def decorator_a(func): print('Get in decorator_a') def inner_a(*args,**kwargs): print('Get in inner_a') res = func(*args,**kwargs) return res return inner_a def decorator_b(func): print('Get in decorator_b') def inner_b(*args,**kwargs): print('Get in inner_b') res = func(*args,**kwargs) return res return inner_b @decorator_b @decorator_a def f(x): print('Get in f') return x * 2 f(1)
輸出結(jié)果如下:
'Get in decorator_a'
'Get in decorator_b'
'Get in inner_b'
'Get in inner_a'
'Get in f'
是不是很像django中的中間件的執(zhí)行順序,其實(shí)原理都差不多。
引用其他博主的描述:
Python裝飾器(decorator)在實(shí)現(xiàn)的時(shí)候,被裝飾后的函數(shù)其實(shí)已經(jīng)是另外一個(gè)函數(shù)了(函數(shù)名等函數(shù)屬性會(huì)發(fā)生改變),為了不影響,Python的functools包中提供了一個(gè)叫wraps的decorator來消除這樣的副作用。寫一個(gè)decorator的時(shí)候,最好在實(shí)現(xiàn)之前加上functools的wrap,它能保留原有函數(shù)的名稱和docstring。
補(bǔ)充:為了訪問原函數(shù)此函數(shù)會(huì)設(shè)置一個(gè)__wrapped__屬性指向原函數(shù), 這樣就可以解釋上面1.3節(jié)中我們的寫法了。
未完待續(xù)。。。。。。。。。
lru_cache緩存裝飾器提供的功能有:
緩存被裝飾對象的結(jié)果(基礎(chǔ)功能)
獲取緩存信息
清除緩存內(nèi)容
根據(jù)參數(shù)變化緩存不同的結(jié)果
LRU算法當(dāng)緩存數(shù)量大于設(shè)置的maxsize時(shí)清除最不常使用的緩存結(jié)果
從列出的功能可知,python自帶的lru_cache緩存方法可以滿足我們?nèi)粘9ぷ髦写蟛糠中枨螅?可是它不包含一個(gè)重要的特性就是,超時(shí)自動(dòng)刪除緩存結(jié)果,所以在我們自制的my_cache中我們將實(shí)現(xiàn)緩存的超時(shí)過期功能。
在作用域內(nèi)存在一個(gè)相對全局的字典變量cache={}
在作用域內(nèi)設(shè)置相對全局的變量包含命中次數(shù) hits,未命中次數(shù) misses ,最大緩存數(shù)量 maxsize和 當(dāng)前緩存大小 currsize
第二點(diǎn)中的緩存信息中增加緩存加入時(shí)間和緩存有效時(shí)間
待實(shí)現(xiàn)。。。。。。。。。。。。
比較類型 | lru_cache | redis |
---|---|---|
緩存類型 | 緩存在app進(jìn)程內(nèi)存中 | 緩存在redis管理的內(nèi)存中 |
分布式 | 只緩存在單個(gè)app進(jìn)程中 | 可做分布式緩存 |
數(shù)據(jù)類型 | hash 參數(shù)作為key,返回結(jié)果為value | 有5種類型的數(shù)據(jù)結(jié)構(gòu) |
適用場景 | 比較小型的系統(tǒng)、單體應(yīng)用 | 常用的緩存解決方案 |
功能 | 緩存功能但是缺少過期時(shí)間控制,但是使用上更加便捷 | 具備緩存需要的各種要素 |
感謝各位的閱讀!關(guān)于“python中自帶緩存lru_cache怎么用”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,讓大家可以學(xué)到更多知識,如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。