leetcode链表之全O(1)的数据结构

432、全O(1)的数据结构
题目:

请你设计一个用于存储字符串计数的数据结构,并能够返回计数最小和最大的字符串。

实现 AllOne 类:

  • AllOne() 初始化数据结构的对象。
  • inc(String key) 字符串 key 的计数增加 1 。如果数据结构中尚不存在 key ,那么插入计数为 1 的 key 。
  • dec(String key) 字符串 key 的计数减少 1 。如果 key 的计数在减少后为 0 ,那么需要将这个 key 从数据结构中删除。测试用例保证:在减少计数前,key 存在于数据结构中。
  • getMaxKey() 返回任意一个计数最大的字符串。如果没有元素存在,返回一个空字符串 “” 。
  • getMinKey() 返回任意一个计数最小的字符串。如果没有元素存在,返回一个空字符串 “” 。

思路:

这道题初看起来有点懵,不知道采用什么数据结构,不过在题目中需要返回计数最大的字符串,和计数最小的字符串,这是不是和之前LRU缓存有点像呢?

不过这里和LRU缓存不同的是,这里每个key都是有一个计数次数的,而不是和LRU缓存一样,直接插入到队列首部。

既然分析这么多,那就开始写代码吧

class DoubleLinkedList:
    def __init__(self, key="", value=None, next=None, prev=None):
        self.key = key
        # 这里的value记录着结点的计数
        self.value = value
        self.next = next
        self.prev = prev

class AllOne:
		# 这里初始化和LRU缓存一样,一个缓存,一个头结点和一个尾结点,这里不需要记录整个缓存的大小。
    def __init__(self):
        self.cache = {}
        self.head = DoubleLinkedList()
        self.tail = DoubleLinkedList()
        self.head.next = self.tail
        self.tail.prev = self.head

    def remove_node(self, node):
        node.prev.next = node.next
        node.next.prev = node.prev

    def move_to_prev(self, node, prev):
        self.remove_node(node)
        self.add_to_prev(node, prev)

    def add_to_prev(self, node, prev):
        node.next = prev.next
        node.prev = prev
        prev.next.prev = node
        prev.next = node
		# 加一操作
    def inc(self, key: str) -> None:
      	# 如果key在缓存中,说明之前已经添加了key,这个时候循环向前遍历,将其向前移动
        if key in self.cache:
            node = self.cache[key]
            count = node.value + 1
            # 更新node的value值
            node.value = count
            cur = node
            while cur.prev:
              	# 此时要考虑,如果要更新的结点,没有比其更大的结点,那么循环遍历会遍历的头结点
                if not cur.prev.value or cur.prev.value >= count:
                    break
                cur = cur.prev
            # 如果当前结点的前一个结点和要新增的结点不是一个的时候,也就是说需要进行结点的移动
            if node.prev != cur.prev:
                self.move_to_prev(node, cur.prev)
        # 如果key不再缓存中,直接插到链表结尾即可
        else:
            node = DoubleLinkedList(key, 1)
            self.add_to_prev(node, self.tail.prev)
            self.cache[key] = node

    def dec(self, key: str) -> None:
        if key not in self.cache: return
        node = self.cache[key]
        count = node.value - 1
        node.value = count
        # 减计数需要考虑是否减到0了,如果减到0,需要删除结点,并从缓存中删除
        if count <= 0:
            self.remove_node(node)
            self.cache.pop(key)
        else:
            cur = node
            # 如果不为0, 那么就需要向后将结点放到合适的位置
            while cur.next:
                if not cur.next.value or  cur.next.value <= count:
                    break
                cur = cur.next
            if cur.next != node.next:
                self.move_to_prev(node, cur)
		#此时getMax就是头部结点的key
    def getMaxKey(self) -> str:
        return self.head.next.key

		# getMin获取尾结点的key
    def getMinKey(self) -> str:
        return self.tail.prev.key
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

溪语流沙

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值