[Leetcode]146. LRU Cache @python

题目

Design and implement a data structure for Least Recently Used (LRU) cache. It should support the following operations: get and set.

get(key) - Get the value (will always be positive) of the key if the key exists in the cache, otherwise return -1.
set(key, value) - Set or insert the value if the key is not already present. When the cache reached its capacity, it should invalidate the least recently used item before inserting a new item.

题目要求

要求实现LRU的缓存机制。LRU机制是在缓存溢出时,删除最近最少使用的缓存元素的机制

解题思路

解这道题,我们要考虑LRU的特性。要知道哪些缓存元素是最近被访问过的,哪些已经很久没有访问过了。链表是可以做这件事情的,但是链表删除和替换位置的操作复杂性为 O(n) ,为了优化效率,采用双向链表来存储,但是双向链表的查找复杂度仍为 O(n) ,这时采用哈希表来达到 O(1) 的访问复杂度。每次访问到一个元素,都将该元素移动到链表的头部。当超过缓存限制时,删除链表尾部的元素,同时从哈希表中删除相应的键值。

代码

class Node:
    def __init__(self,key,val):
        self.val = val
        self.key = key
        self.next = None
        self.pre = None

class DoubleLinkedList:
    def __init__(self):
        self.header = None
        self.tail = None

    def add_first(self,node):
        if self.header == None:
            self.header = node
            self.tail = node
        else:
            h = self.header
            self.header = node
            node.next = h
            h.pre = node
    def remove(self,node):
        pre,next = node.pre,node.next
        if pre:
            pre.next = next
        else:
            self.header = next
        if next:
            next.pre = pre
        else:
            self.tail = pre
        node.pre = None
        node.next = None
    def remove_last(self):
        if self.tail == None:
            return None
        tail = self.tail
        self.remove(tail)
        return tail

class LRUCache(object):

    def __init__(self, capacity):
        """
        :type capacity: int
        """
        self.capacity = capacity
        self.count = 0
        self.linkedlist = DoubleLinkedList()
        self.maps = {}

    def get(self, key):
        """
        :rtype: int
        """
        if key not in self.maps:
            return -1
        node = self.maps[key]
        self.linkedlist.remove(node)
        self.linkedlist.add_first(node)
        return node.val

    def set(self, key, value):
        """
        :type key: int
        :type value: int
        :rtype: nothing
        """
        if key not in self.maps:
            self.count += 1
            node = Node(key,value)
            self.maps[key] = node
            self.linkedlist.add_first(node)
        else:
            node = self.maps[key]
            node.val = value
            self.linkedlist.remove(node)
            self.linkedlist.add_first(node)
        if self.count > self.capacity:
            self.count -= 1
            tail = self.linkedlist.remove_last()
            del self.maps[tail.key]

你好!对于LeetCode上的问题994.腐烂的橘子,你可以使用Python来解决。下面是一个示例代码: ```python from collections import deque def orangesRotting(grid): # 记录网格的行数和列数 row, col = len(grid), len(grid[0]) # 定义四个方向:上、下、左、右 directions = [(-1, 0), (1, 0), (0, -1), (0, 1)] # 使用队列来保存腐烂的橘子的位置 queue = deque() # 记录新鲜橘子的数量 fresh_count = 0 # 遍历整个网格,初始化队列和新鲜橘子的数量 for i in range(row): for j in range(col): if grid[i][j] == 2: # 腐烂的橘子 queue.append((i, j)) elif grid[i][j] == 1: # 新鲜橘子 fresh_count += 1 # 如果新鲜橘子的数量为0,直接返回0 if fresh_count == 0: return 0 # 初始化分钟数 minutes = 0 # 开始进行BFS,直到队列为空 while queue: # 记录当前分钟数下,队列中的元素数量 size = len(queue) # 遍历当前分钟数下的所有腐烂的橘子 for _ in range(size): x, y = queue.popleft() # 遍历四个方向 for dx, dy in directions: nx, ny = x + dx, y + dy # 判断新位置是否在网格内,并且是新鲜橘子 if 0 <= nx < row and 0 <= ny < col and grid[nx][ny] == 1: # 将新鲜橘子变为腐烂状态 grid[nx][ny] = 2 # 将新鲜橘子的位置加入队列 queue.append((nx, ny)) # 新鲜橘子的数量减1 fresh_count -= 1 # 如果当前分钟数下,没有新鲜橘子了,结束循环 if fresh_count == 0: break # 每遍历完一层,分钟数加1 minutes += 1 # 如果最后还有新鲜橘子,返回-1,否则返回分钟数 return -1 if fresh_count > 0 else minutes ``` 你可以将给定的网格作为参数传递给`orangesRotting`函数来测试它。请注意,该代码使用了BFS算法来遍历橘子,并计算腐烂的分钟数。希望能对你有所帮助!如果有任何疑问,请随时问我。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值