Python實(shí)現(xiàn)LRU算法
在第一節(jié)中已經(jīng)實(shí)現(xiàn)了雙向鏈表DoubleLinkedList類,本節(jié)我們基于雙向鏈表實(shí)現(xiàn)LRU(Least Recently Used最近最少使用)緩存置換算法。Redis的淘汰機(jī)制就包括LRU算法,用來淘汰那些最近最少使用的數(shù)據(jù),具體怎么使用可在redis的配置文件中設(shè)置。
一、LRU算法的實(shí)現(xiàn)
邏輯很簡單,get和put兩種操作,其中g(shù)et時(shí)如果元素存在則將節(jié)點(diǎn)從當(dāng)前位置移到鏈表頭部,表示最近被訪問到的節(jié)點(diǎn);put時(shí)也是,不管節(jié)點(diǎn)之前存不存在都要移動到鏈表頭部。同樣通過一個(gè)map來實(shí)現(xiàn)查找時(shí)的O(1)復(fù)雜度。
class LRUCache(object):
? ? def __init__(self, capacity=0xffffffff):
? ? ? ? """
? ? ? ? LRU緩存置換算法 最近最少使用
? ? ? ? :param capacity:
? ? ? ? """
? ? ? ? self.capacity = capacity
? ? ? ? self.size = 0
? ? ? ? self.map = {}
? ? ? ? self.list = DoubleLinkedList(capacity)
? ? def get(self, key):
? ? ? ? """
? ? ? ? 獲取元素
? ? ? ? ? ? 獲取元素不存在 返回None
? ? ? ? ? ? 獲取元素已存在 將節(jié)點(diǎn)從當(dāng)前位置刪除并添加至鏈表頭部
? ? ? ? :param key:
? ? ? ? :return:
? ? ? ? """
? ? ? ? # 元素不存在
? ? ? ? if key not in self.map:
? ? ? ? ? ? return None
? ? ? ? node = self.map.get(key)
? ? ? ? self.list.remove(node)
? ? ? ? self.list.append_front(node)
? ? ? ? return node.value
? ? def put(self, key, value):
? ? ? ? """
? ? ? ? 添加元素
? ? ? ? ? ? 被添加的元素已存在 更新元素值并已到鏈表頭部
? ? ? ? ? ? 被添加的元素不存在
? ? ? ? ? ? ? ? 鏈表容量達(dá)到上限 刪除尾部元素
? ? ? ? ? ? ? ? 鏈表容量未達(dá)上限 添加至鏈表頭部
? ? ? ? :param key:
? ? ? ? :param value:
? ? ? ? :return:
? ? ? ? """
? ? ? ? if key in self.map:
? ? ? ? ? ? node = self.map.get(key)
? ? ? ? ? ? node.value = value
? ? ? ? ? ? self.list.remove(node)
? ? ? ? ? ? self.list.append_front(node)
? ? ? ? else:
? ? ? ? ? ? if self.size >= self.capacity:
? ? ? ? ? ? ? ? old_node = self.list.remove()
? ? ? ? ? ? ? ? del self.map[old_node.key]
? ? ? ? ? ? ? ? self.size -= 1
? ? ? ? ? ? node = Node(key, value)
? ? ? ? ? ? self.map[key] = node
? ? ? ? ? ? self.list.append_front(node)
? ? ? ? ? ? self.size += 1
? ? ? ? return node
? ? def print(self):
? ? ? ? """
? ? ? ? 打印當(dāng)前鏈表
? ? ? ? :return:
? ? ? ? """
? ? ? ? self.list.print()二、測試邏輯
if __name__ == '__main__': ? ? lru_cache = LRUCache(3) ? ? lru_cache.put(1, 1) ? ? lru_cache.print() ? ? lru_cache.put(2, 2) ? ? lru_cache.print() ? ? print(lru_cache.get(1)) ? ? lru_cache.print() ? ? lru_cache.put(3, 3) ? ? lru_cache.print() ? ? lru_cache.put(1, 100) ? ? lru_cache.print() ? ? lru_cache.put(4, 4) ? ? lru_cache.print() ? ? print(lru_cache.get(1)) ? ? lru_cache.print()
測試結(jié)果:

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
教你用python實(shí)現(xiàn)一個(gè)無界面的小型圖書管理系統(tǒng)
今天帶大家學(xué)習(xí)怎么用python實(shí)現(xiàn)一個(gè)無界面的小型圖書管理系統(tǒng),文中有非常詳細(xì)的圖文解說及代碼示例,對正在學(xué)習(xí)python的小伙伴們有很好地幫助,需要的朋友可以參考下2021-05-05
python調(diào)用機(jī)器喇叭發(fā)出蜂鳴聲(Beep)的方法
這篇文章主要介紹了python調(diào)用機(jī)器喇叭發(fā)出蜂鳴聲(Beep)的方法,實(shí)例分析了Python調(diào)用winsound模塊的使用技巧,需要的朋友可以參考下2015-03-03
python疲勞駕駛困倦低頭檢測功能的實(shí)現(xiàn)
這篇文章主要介紹了python疲勞駕駛困倦低頭檢測,該系統(tǒng)可以檢測一個(gè)人在開車時(shí)是否困倦,及時(shí)提醒,做到安全隱患排查,對實(shí)現(xiàn)代碼感興趣的朋友一起看看吧2022-04-04

