TikTok真题第6天 | 146.LRU缓存、333.最大的二分搜索树、621.任务调度器

146.LRU缓存

题目链接:146.lru-cache

解法:

这个题代码量大,光看题解就1个小时多了,看完写下来花了两小时多...

使用哈希表+双向链表来实现LRU缓存的特性,即哈希表可以实现get为O(1)复杂度,双向链表可以实现put、remove都是O(1)的复杂度。

代码实现时,这里双向链表,靠近头部的键值对是最久未使用的,而靠近尾部的键值对是最近使用的。也有的实现,头部放最近使用的,尾部放最久未使用。

而删除最久未使用的node时,需要在哈希表和双向链表中都删除,那么可以由双向链表得到最久未使用的node,从而得到node的key,再从哈希表中删除,这是哈希表和双向链表之间的联系。

一个技巧是使用dummy虚拟节点,有的题解用两个dummy表示首尾节点,有的题解只用了一个dummy。个人认为写法要容易理解,简洁是其次的要求。两个dummy的写法更好理解。

题解参考labuladong,代码多一些,但是很清晰,然后翻译成了C++版本:labuladong,LRU缓存

边界条件:无

时间复杂度:对于 put 和 get 都是 O(1)

空间复杂度:O(capacity)

class Node {
public:int key, val;Node *pre, *next;Node (int k=0, int v=0): key(k), val(v), pre(nullptr), next(nullptr) {};
};class DoubleList {
private:Node *head, *tail;int size;public:DoubleList() {head = new Node();tail = new Node();head->next = tail;tail->pre = head;size = 0;}int getSize() {return size;}// 末尾的node是最近使用的void addLast(Node* x) {x->next = tail;x->pre = tail->pre;x->next->pre = x;x->pre->next = x;size++;}void remove(Node* x) {x->pre->next = x->next;x->next->pre = x->pre;size--;}// 开头的node是最久未使用的,返回的目的是为了在map中删除Node* removeFirst() {if (head->next == tail) return nullptr;Node* first = head->next;remove(first);return first;}
};class LRUCache {
private:unordered_map<int, Node*> map;DoubleList cache;int cap;private:// 用于get时,把node作为最近使用的void makeRecently (int key) {Node* node = map[key];cache.remove(node);cache.addLast(node);}void addRecently (int key, int val) {Node* node = new Node(key, val);cache.addLast(node);map[key] = node;}void deleteKey (int key) {Node* node = map[key];cache.remove(node);map.erase(key);delete node;}void removeLeastRecently () {Node* node = cache.removeFirst();int key = node->key;map.erase(key);delete node;}public:LRUCache(int capacity): cap(capacity){};int get(int key) {// 如果不存在,则返回-1if (map.find(key) == map.end()) return -1;makeRecently(key);return map[key]->val;}void put(int key, int value) {// 如果存在,则先删除,再加入if (map.find(key) != map.end()) {deleteKey(key);addRecently(key, value);return;}// 如果不存在,且空间满了,则先移除最久未使用if (cap == cache.getSize()) {removeLeastRecently();}addRecently(key, value);}
};

333.最大的二分搜索树

题目链接:333.largest-bst-subtree

解法:

参考题解:https://www.cnblogs.com/grandyang/p/5188938.html

这个题有follow up,那么首先给出O(n^2)的解法。

每个节点都当做根节点,来验证其是否是二叉搜索数。如果当前node是二叉搜索树,就记录节点的个数,并返回节点个数,如果不是二叉搜索树,那就验证左右子树是否为二叉搜索树,分别记录节点的个数,然后取二者中的最大值进行返回。

思路可以说是DFS和分而治之(左子树和右子树分别进行DFS)。

由于每个节点都要对树遍历一次,所以时间复杂度为O(n^2)。

下面是O(n)复杂度的解法。

只允许遍历一次整个二叉树,由于满足要求的二叉搜索子树必定是有叶节点的,所以思路就是先递归到最左子节点,然后逐层往上递归。对于每一个节点,都记录当前最大的 BST 的节点数,当做为左子树的最大值,和做为右子树的最小值。

当每次遇到左子节点不存在或者当前节点值大于左子树的最大值,且右子树不存在或者当前节点值小于右子树的最小数时,说明 BST 的节点数又增加了一个,更新结果及其参数。

如果当前节点不是 BST 的节点,那么更新 BST 的节点数 res 为左右子节点的各自的 BST 的节点数的较大值。

这个O(n)的解法实在看得眼花缭乱,这次也没有理解得很好。下次再细看了。

边界条件:无

时间复杂度:O(n)

空间复杂度:O(h),树的深度

// O(n^2)的解法
class Solution {
public:int largestBSTSubtree(TreeNode* root) {if (!root) return 0;// 如果root是BST,那么左右子树一定是,但root一定是比左右子树更大的BST,所以直接returnif (isValid(root, INT_MIN, INT_MAX)) return count(root);return max(largestBSTSubtree(root->left), largestBSTSubtree(root->right));}bool isValid(TreeNode* root, int min, int max) {if (!root) return true;if (root->val <= min || root->val >= max) return false;// 该节点满足,那么继续看左右节点是否满足return isValid(root->left, min, root->val) && isValid(root->right, root->val, max);}int count(TreeNode* node) {if (!node) return 0;return count(node->left) + count(node->right) + 1;}
};
/*** Definition for a binary tree node.* struct TreeNode {*     int val;*     TreeNode *left;*     TreeNode *right;*     TreeNode() : val(0), left(nullptr), right(nullptr) {}*     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}*     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}* };*/
// O(n)的解法
class Solution {
public:int largestBSTSubtree(TreeNode* root) {// res中有三个元素: 以当前结点为根结点的树的最小值,最大值,最大的 BST 子树的结点个数vector<int> res = helper(root);return res[2];}vector<int> helper(TreeNode* node) {if (!node) return {INT_MAX, INT_MIN, 0};vector<int> left = helper(node->left), right = helper(node->right);// 大于左子树的最大值,小于右子树的最小值,那么是BSTif (node->val > left[1] && node->val < right[0]) {return {min(node->val, left[0]), max(node->val, right[1]), left[2] + right[2] + 1};} else {// 这里不太好理解,这是用于破坏BST规则,node->val > left[1] && node->val < right[0]return {INT_MIN, INT_MAX, max(left[2], right[2])};}}
};

621.任务调度器

题目链接:task-scheduler

解法:

这种题真是奇技淫巧,思想确实精妙,但是属于脑筋急转弯类型。吐槽一下,刷这些题真是浪费青春,造孽啊!

没啥好说的,直接参考题解:桶思想

边界条件:无

时间复杂度:O(nlogn),排序

空间复杂度:O(1)

class Solution {
public:int leastInterval(vector<char>& tasks, int n) {// 总的任务数int len = tasks.size();// 统计每个任务的数量vector<int> vec(26);for (char c: tasks) {vec[c-'A']++;}// 按任务数量进行降序排列,任务是啥不重要了sort(vec.begin(), vec.end(), [](int& x, int&y) {return x > y;});// 统计任务数量最多,且数量相等的任务,有多少个int cnt = 1;while (cnt < vec.size() && vec[cnt] == vec[0]) {cnt++;}return max(len, (vec[0]-1)*(n+1)+cnt);}
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/582104.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Mybatis Mapper XML文件-缓存(cache)

MyBatis包含一个强大的事务查询缓存特性&#xff0c;可以进行灵活的配置和自定义。在MyBatis 3的缓存实现中进行了许多改进&#xff0c;使其更加强大且更易于配置。 默认情况下&#xff0c;仅启用了本地会话缓存&#xff0c;该缓存仅用于缓存会话期间的数据。要启用全局的第二…

C++ 一个有bug的贪吃蛇。。。。。。。。

C 一个有bug的贪吃蛇。。。。。。。。 #include <graphics.h> #include<Windows.h> #include<Mmsystem.h> #include<conio.h> #include<time.h> #include<stdio.h> #include<easyx.h> using namespace std; #pragma warning(di…

Qt之自定义分页(翻页)控件

当数据量较大时,分页显示是个不错的选择。这里用百家姓来演示分页效果,包括首页、上一页、下一页、尾页和跳转。 一.效果 每页15个姓氏。 二.实现 QHPageWidget.h #ifndef QHPAGEWIDGET_H #define QHPAGEWIDGET_H#include <QWidget> #include <QStandardItemMod…

DBeaver连接hive

1.新建hive连接 其中主机填写hive所在节点地址&#xff0c;端口10000为默认&#xff0c;数据库名不填则是默认default数据库&#xff0c;用户名密码填写hadoop集群中能操作hdfs的用户和密码。 2.编辑驱动&#xff0c;驱动的jar包从安装的hive下的jdbc路径下获取&#xff0c;例…

第8章 社会工程学攻击

社会工程学是利用人性弱点体察、获取有价值信息的实践方法&#xff0c;它是一种欺骗的艺术。在缺少目标系统的必要信息时&#xff0c;社会工程学技术是渗透测试人员获取信息的至关重要的手段。对所有类型的组织&#xff08;单位&#xff09;而言&#xff0c;人都是安全防范措施…

【flink番外篇】9、Flink Table API 支持的操作示例(3)- 通过API查询表和使用窗口函数的查询

Flink 系列文章 一、Flink 专栏 Flink 专栏系统介绍某一知识点&#xff0c;并辅以具体的示例进行说明。 1、Flink 部署系列 本部分介绍Flink的部署、配置相关基础内容。 2、Flink基础系列 本部分介绍Flink 的基础部分&#xff0c;比如术语、架构、编程模型、编程指南、基本的…

软件开发者必读!2024年30大趋势提前曝光!

软件是科技行业的基础。从智能手机到复杂的企业系统&#xff0c;软件是推动这些技术的核心。在探索数字时代的过程中&#xff0c;了解软件开发的趋势非常重要。人工智能、机器学习、物联网等技术的快速发展对软件开发行业产生了巨大影响。本文旨在通过30个数据汇总&#xff0c;…

STM32 基础知识(探索者开发板)--93讲 PWM

预分频器相当于一个计数器&#xff0c;2分频就是接收2个脉冲传递一个脉冲&#xff0c;3分频就是接收3个脉冲传递一个脉冲&#xff0c;最高65535分频&#xff0c;那么总计时间能达到65535*65535*1/72MHZ 约59秒&#xff0c;没有分频器只能计数最高0.09秒 PWM配置步骤 1.配置定时…

《长江丛刊》期刊投稿发表

《长江丛刊》杂志是由国家新闻出版总署批准的正规期刊&#xff0c;是一本文学类综合性刊物&#xff0c;包括文学评论以及文学作品的发表。是广大专家、学者、教师、学子发表论文、交流信息的重要平台&#xff0c;符合评职称要求&#xff08;单位有特殊要求除外&#xff09;。 …

docker小白第八天

docker小白第八天 redis常规安装 前期已经配好了阿里云容器镜像加速库 docker search redis docker pull redis:6.0.8 docker images redis:6.0.8启动容器,并进入容器的命令行界面 docker run -d -p 6379:6379 redis:6.0.8 docker ps docker exec -it 容器id /bin/bash验证…

可替代 Spring,国产 Solon v2.6.3 发布

Solon 是什么框架&#xff1f; Java 新的"生态级"应用开发框架。从零开始构建&#xff0c;有自己的标准规范与开放生态&#xff08;历时六年&#xff0c;具备全球第二级别的生态规模&#xff09;。 相对于 Spring&#xff0c;有什么特点&#xff1f; 国产&#xf…

Java生态系统的进化:从JDK 1.0到今天

目录 前言 JDK 1.0&#xff1a;开启Java时代 JDK 1.1&#xff1a;Swing和内部类 JDK 1.2&#xff1a;Collections框架和JIT编译器 JDK 1.5&#xff1a;引入泛型和枚举 JDK 1.8&#xff1a;Lambda表达式和流 JDK 11以后&#xff1a;模块化和新特性 未来展望 总结 作者简…

Abaqus许可证有效期及续期方法

作为工程设计和仿真领域的知名软件&#xff0c;Abaqus受到广大用户青睐。然而&#xff0c;许可证有效期的问题常常困扰着用户。为了帮助您更好地了解Abaqus许可证有效期及续期方法&#xff0c;我们特别邀请了行业专家进行详细解读。 一、Abaqus许可证有效期 Abaqus许可证的有…

【2023-12-23周报】

2023-12-23周报 本周开展工作下周工作计划 本周开展工作 上线工作小程序的水印添加以及删除项目的合并以及调整更改后的项目接口上周由于我们的后端不在&#xff0c;所以对于我们现在的小程序项目进行了一些深入的思考这周的话和后端进行联调了后端的接口 上线的话我们为小程…

Elasticsearch 常用 REST API 之集群APIs

Cluster health API 集群运行状况API返回关于集群运行状况的简单状态。您还可以使用API仅获取指定数据流和索引的健康状态。对于数据流&#xff0c;API 检索流的支持索引的运行状况。 集群健康状态为:绿色、黄色和红色。在shard级别&#xff0c;红色状态表示集群中没有分配特…

高并发下数据库插入操作死锁问题

1. 问题&#xff1a; 项目中出现如下报错&#xff1a; org.springframework.dao.DeadlockLoserDataAccessException: com.xxxMapper.insert (batch index #1) failed. Cause: java.sql.BatchUpdateException: Deadlock found when trying to get lock; try restarting trans…

Transformer实现的一个Demo

RT&#xff0c;直接上代码&#xff0c;可以跑通&#xff1a; #encoding:utf-8 import torch import torch.nn as nn import numpy as np import math class Config(object): def __init__(self): self.vocab_size 6 self.d_model 512 self.n_heads 4 assert self.d_model…

UI自动化Selenium 元素定位之Xpath

一、元素定位方式 selenium中定位元素&#xff0c;通常有几种方式&#xff1a; 1、通过id定位&#xff1a;By.ID 2、通过Name定位&#xff1a;By.Name 3、通过元素其他属性定位&#xff0c;如class、type、text文本。。。。。。等等&#xff0c;如果要用属性定位那就需要使…

图论 经典例题

1 拓扑排序 对有向图的节点排序&#xff0c;使得对于每一条有向边 U-->V U都出现在V之前 *有环无法拓扑排序 indegree[], nxs[];//前者表示节点 i 的入度&#xff0c;后者表示节点 i 指向的节点 queue [] for i in range(n):if indege[i] 0: queue.add(i)// 入度为0的节…

虚析构和纯虚析构

多态使用时&#xff0c;如果子类中有属性开辟到堆区&#xff0c;那么父类的指针在释放时无法调用到子类的析构代码 解决方式&#xff1a;将父类中的析构代码函数改为虚析构或者纯虚析构 虚析构和纯虚析构共性&#xff1a; 可以解决父类指针释放子类对象 都需要有具体的函数…