python實(shí)現(xiàn)類redis緩存

越來越覺得的緩存是計(jì)算機(jī)科學(xué)里最NB的發(fā)明(沒有之一), 現(xiàn)在項(xiàng)目用的是redis做的緩存, 它的兩個(gè)特性用的蠻順手的:

  1. 鍵值查找功能
  2. 緩存可設(shè)置過期時(shí)間

突突然的凯肋,覺得用python也可以簡單的模擬一下,做一個(gè)本地的輕量級(jí)緩存.(不過, 注意一點(diǎn):redis的緩存可以用于分布式, python模擬的則不行, 但是如果把本地緩存的過期時(shí)間設(shè)的短一點(diǎn),比如10s, 在大并發(fā)下還是有不錯(cuò)表現(xiàn)的)

對(duì)于鍵值查找功能, python原生的字典dict完美勝任. 對(duì)于緩存自動(dòng)過期, 簡單的想法后臺(tái)起個(gè)服務(wù), 定期檢測, 但是這樣代碼會(huì)相對(duì)復(fù)雜莉撇,而且還要搶占寶貴的cpu資源诈皿,失去了輕量級(jí)的初衷.

那么思路是這樣的:

  1. 只要將每個(gè)對(duì)應(yīng)的緩存鍵值蚤霞,加一個(gè)expire字段,代表過期時(shí)間點(diǎn)昔馋,
  • 第一次獲取, expire=當(dāng)前時(shí)間點(diǎn)+過期時(shí)間
  • 非第一次獲取時(shí), 通過expire判斷是否已過期, 如果過期則可認(rèn)為數(shù)據(jù)沒找到五芝,反之返回正常的緩存
    簡單的代碼如下:
   def get(self, key):

        value = self._cache.get(key, self.notFound)

        if(value is not self.notFound):

            expire = value[r'expire']

            if( self.nowTime() > expire):
                return self.notFound
            else:
                return value

        else:
            return self.notFound
  1. 通過第1步, 基本功能已實(shí)現(xiàn)刊咳,現(xiàn)在收拾"爛攤子". 步驟1明顯的確點(diǎn)就是內(nèi)存泄漏了(搞python的應(yīng)該不怎么聽的到"內(nèi)存泄漏"),原因很明顯彪见,通過字典模擬redis緩存, 所有數(shù)據(jù)都保存在緩存字典中(即使它過期了), 因?yàn)闆]人去刪除它們.

解決的方法是娱挨,采用弱引用
python 文檔有有句:

A primary use for weak references is to implement caches or mappings holding large objects, where it’s desired that a large object not be kept alive solely because it appears in a cache or mapping.

就是說余指,弱引用主要的用途也是為了實(shí)現(xiàn)緩存.

對(duì)于我們的應(yīng)用,WeakValueDictionary這貨就很符合需求跷坝,那么代碼就如下這樣了:

class LocalCache():

    notFound = object()

    class Dict(dict):

        def __del__(self):
            pass

    def __init__(self, maxlen=2):

        self.weak = weakref.WeakValueDictionary()

    @staticmethod
    def nowTime():
        return int(time.time())

    def get(self, key):

        value = self.weak.get(key, self.notFound)

        if(value is not self.notFound):
            expire = value[r'expire']

            if( self.nowTime() > expire):
                return self.notFound
            else:
                return value

        else:
            return self.notFound

    def set(self, key, value):

        self.weak[key] = LocalCache.Dict(value)

幾點(diǎn)說明下:

  • 創(chuàng)建內(nèi)部類Dict的原因
    python文檔的說法
Several built-in types such as list and dict do not directly support weak references but can add support through subclassing:
classDict(dict):
    pass
obj=Dict(red=1,green=2,blue=3)

就是說內(nèi)建的list和dict不能直接支持弱引用酵镜,但是繼承他們的子類就支持弱引用了. so..

  • weakref.WeakValueDictionary說明
    現(xiàn)在就是用WeakValueDictionary來代替之前的dict了,弱引用的優(yōu)勢就是:
    WeakValueDictionary隨時(shí)都可能被回收(聽上去很不靠譜,不過下面有解決方法), 所以不用擔(dān)心之前的內(nèi)存泄漏問題.
    可能有人擔(dān)心,WeakValueDictionary只是value值是弱引用柴钻,也就說value可以被回收淮韭,但是key值回一直存在,導(dǎo)致泄漏,don't wrong, 有文檔為證:
Entries in the dictionary will be discarded when no strong reference to the value exists any more.

即如果value值沒有強(qiáng)引用了贴届,那么對(duì)應(yīng)的記錄就會(huì)被丟棄(這句話有彩蛋). 所以也不用擔(dān)心key導(dǎo)致的泄漏了.

當(dāng)初寫完這個(gè)后靠粪,感覺一切都比較的perfect. 但是還是too young too simple.

  1. 最后較成熟的方案

重新關(guān)注下第二步, 有彩蛋的那句話, 這句話再深入理解下就是,如果self.weak[key] = LocalCache.Dict(value)這樣毫蚓,LocalCache.Dict(value)沒有其他強(qiáng)引用, 那么對(duì)不起占键,下一瞬間這個(gè)記錄就沒了(WTF).

所以之前的寫法會(huì)導(dǎo)致--沒有任何緩存作用(如果你耐著性子看到這,估計(jì)要罵娘了元潘。畔乙。。)柬批,不過既然都寫了這么多啸澡,方案還是有的(我在 http://stackoverflow.com 找到類似的方案,稍微改進(jìn)了下)氮帐,既然要強(qiáng)引用嗅虏,那就給他強(qiáng)引用了,代碼如下:

import weakref, collections
import time

class LocalCache():

    notFound = object()

    class Dict(dict):

        def __del__(self):
            pass

    def __init__(self, maxlen=10):

        self.weak = weakref.WeakValueDictionary()

        self.strong = collections.deque(maxlen=maxlen)

    @staticmethod
    def nowTime():

        return int(time.time())

    def get(self, key):

        value = self.weak.get(key, self.notFound)

        if(value is not self.notFound):

            expire = value[r'expire']

            if( self.nowTime() > expire):

                return self.notFound

            else:

                return value

        else:

            return self.notFound

    def set(self, key, value):

        self.weak[key] = strongRef = LocalCache.Dict(value)

        self.strong.append(strongRef)

代碼跟之前的差不多上沐,就是多了self.strong這個(gè)隊(duì)列來保存強(qiáng)引用, 并利用collections.deque的一個(gè)特性:

the deque is bounded to the specified maximum length. Once a bounded length deque is full, when new items are added, a corresponding number of items are discarded from the opposite end.

意思是如果給deque設(shè)置了大小(通過maxlen皮服,不傳或設(shè)為None則沒有限制), 那么deque滿的時(shí)候,新添加的對(duì)象會(huì)將之前的'老家伙擠出去'.

整個(gè)過程是這樣的,剛開始往緩存加數(shù)據(jù)時(shí), 添加的每一個(gè)值都有一個(gè)弱引用和強(qiáng)引用龄广,不停的加硫眯,直到deque的隊(duì)列滿了(地主家也沒余糧啊), 這時(shí)后面每加一個(gè)择同,都將導(dǎo)致deque中最早加入的強(qiáng)引用被deque廢棄两入,而被廢棄的強(qiáng)引用對(duì)應(yīng)的值只有弱引用了,于是與之相關(guān)的WeakValueDictionary記錄也被回收了(不知道有沒人閃過虛擬內(nèi)存中內(nèi)存不夠用時(shí)敲才,數(shù)據(jù)在硬盤和內(nèi)存搗鼓的畫面)

到這基本已經(jīng)寫完裹纳。當(dāng)然這個(gè)LocalCache類還有很多可完善的地方,這里只是講解下它的形成過程紧武。

  1. 附加一個(gè)應(yīng)用這個(gè)LocalCache類的函數(shù)調(diào)用緩存裝飾器
from functools import wraps
def funcCache(expire=0):

    caches = LocalCache()

    def _wrappend(func):

        @wraps(func)
        def __wrapped(*args , **kwargs):
            #計(jì)算出緩存的key值
            key = str(func) + str(args) + str(kwargs)

            result = caches.get(key)

            if(result is LocalCache.notFound):

                result = func(*args, **kwargs)

                caches.set(key, {r'result':result, r'expire':expire + caches.nowTime() })

                result = caches.get(key)

            return result
                
        return __wrapped

    return _wrappend
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末剃氧,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子阻星,更是在濱河造成了極大的恐慌朋鞍,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,386評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件妥箕,死亡現(xiàn)場離奇詭異滥酥,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)畦幢,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門恨狈,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人呛讲,你說我怎么就攤上這事》捣睿” “怎么了贝搁?”我有些...
    開封第一講書人閱讀 164,704評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長芽偏。 經(jīng)常有香客問我雷逆,道長,這世上最難降的妖魔是什么污尉? 我笑而不...
    開封第一講書人閱讀 58,702評(píng)論 1 294
  • 正文 為了忘掉前任膀哲,我火速辦了婚禮,結(jié)果婚禮上被碗,老公的妹妹穿的比我還像新娘某宪。我一直安慰自己,他們只是感情好锐朴,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,716評(píng)論 6 392
  • 文/花漫 我一把揭開白布兴喂。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪衣迷。 梳的紋絲不亂的頭發(fā)上畏鼓,一...
    開封第一講書人閱讀 51,573評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音壶谒,去河邊找鬼云矫。 笑死,一個(gè)胖子當(dāng)著我的面吹牛汗菜,可吹牛的內(nèi)容都是我干的让禀。 我是一名探鬼主播,決...
    沈念sama閱讀 40,314評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼呵俏,長吁一口氣:“原來是場噩夢啊……” “哼堆缘!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起普碎,我...
    開封第一講書人閱讀 39,230評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤吼肥,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后麻车,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體缀皱,經(jīng)...
    沈念sama閱讀 45,680評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,873評(píng)論 3 336
  • 正文 我和宋清朗相戀三年动猬,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了啤斗。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,991評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡赁咙,死狀恐怖钮莲,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情彼水,我是刑警寧澤崔拥,帶...
    沈念sama閱讀 35,706評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站凤覆,受9級(jí)特大地震影響链瓦,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜盯桦,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,329評(píng)論 3 330
  • 文/蒙蒙 一慈俯、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧拥峦,春花似錦贴膘、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,910評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽揪胃。三九已至,卻和暖如春氛琢,著一層夾襖步出監(jiān)牢的瞬間喊递,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,038評(píng)論 1 270
  • 我被黑心中介騙來泰國打工阳似, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留骚勘,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,158評(píng)論 3 370
  • 正文 我出身青樓撮奏,卻偏偏與公主長得像俏讹,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子畜吊,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,941評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • 分布式緩存技術(shù)PK:選擇Redis還是Memcached玲献? 經(jīng)平臺(tái)同意授權(quán)轉(zhuǎn)載 作者:田京昆(騰訊后臺(tái)研發(fā)工程師)...
    meng_philip123閱讀 68,926評(píng)論 7 60
  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉殉疼,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 1,715評(píng)論 0 9
  • // +-----------------------------------------------------...
    Robinbing閱讀 1,434評(píng)論 0 0
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)捌年,斷路器瓢娜,智...
    卡卡羅2017閱讀 134,657評(píng)論 18 139
  • 今天實(shí)現(xiàn)了一個(gè)愿望。 我和魏先生都喜歡玩網(wǎng)絡(luò)游戲礼预,從大學(xué)到現(xiàn)在一直都喜歡打英雄聯(lián)盟眠砾。以前我們就幻想有一天,我們買了...
    小菠蘿的日常閱讀 177評(píng)論 0 0