力扣面试经典算法150题:O(1) 时间插入、删除和获取随机元素

2024-08-27 04:04

本文主要是介绍力扣面试经典算法150题:O(1) 时间插入、删除和获取随机元素,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

O(1) 时间插入、删除和获取随机元素

今天的题目是力扣面试经典150题中的数组的中等难度题: O(1) 时间插入、删除和获取随机元素。

题目链接:https://leetcode.cn/problems/insert-delete-getrandom-o1/description/?envType=study-plan-v2&envId=top-interview-150

问题描述

实现RandomizedSet 类:

RandomizedSet() 初始化 RandomizedSet 对象
bool insert(int val) 当元素 val 不存在时,向集合中插入该项,并返回 true ;否则,返回 false 。
bool remove(int val) 当元素 val 存在时,从集合中移除该项,并返回 true ;否则,返回 false 。
int getRandom() 随机返回现有集合中的一项(测试用例保证调用此方法时集合中至少存在一个元素)。每个元素应该有 相同的概率 被返回。
你必须实现类的所有函数,并满足每个函数的 平均 时间复杂度为 O(1) 。

  • 示例

    • 输入:
      [“RandomizedSet”, “insert”, “remove”, “insert”, “getRandom”, “remove”, “insert”, “getRandom”]
      [[], [1], [2], [2], [], [1], [2], []]
    • 输出:
      [null, true, false, true, 2, true, false, 2]
  • 解释:
    RandomizedSet randomizedSet = new RandomizedSet();
    randomizedSet.insert(1); // 插入 1,返回 true 表示成功插入
    randomizedSet.remove(2); // 尝试移除 2,返回 false 表示未找到
    randomizedSet.insert(2); // 插入 2,返回 true 表示成功插入
    randomizedSet.getRandom(); // 随机返回 1 或 2,这里假设返回 2
    randomizedSet.remove(1); // 移除 1,返回 true 表示成功移除
    randomizedSet.insert(2); // 尝试插入 2,返回 false 表示已存在
    randomizedSet.getRandom(); // 随机返回 2,因为现在集合中只有 2

题目分析

题目要求我们实现一个类,这个类可以写入,删除或随机获取某个元素,并且要求每个函数的平均时间复杂度为O(1)。

不要被要实现一个类迷惑,其实就是要求我们设计一个数据结构和函数算法,并且要求函数的时间复杂度在规定范围之内就行。

为了实现在 O(1) 时间内完成插入、删除和获取随机元素的操作,我们需要设计一种数据结构,能够快速定位元素的位置,并且在删除元素时不影响随机获取的性能。

解题思路

这个题目需要通过组合我们常用的数据结构来实现需求,考验的就是我们对常见的数据结构的理解与掌握。毕竟是中等难度的题目,能够正确理解掌握现有的数据结构并通过组合实现特定场景的优秀算法,已经是很不错了。

我们先梳理一下常见的数据结构:数组,链表,哈希表,队列,堆,栈。这些数据结构各有各的特点,这里不一一细讲,这东西细讲起来能不是一下两下能讲完的,目前我们需要知道的就是,想完成题目要求,我们需要在这些数据结构中组合使用。

看看题目中的要求,不存在时才能新增,存在时才能删除。也就是说,在执行这两个函数的时候,我们需要先知道元素在不在我们设定的数据结构中。

这个时候,哈希表说:老弟,这个我熟啊!

但是哈希表的新增与删除通常情况下是符合条件的,但是极端情况下是O(n)。此外,获取随机元素通常我们需要遍历整个哈希表才能实现,这个时间复杂度是O(n)。

假如我们在哈希表的基础上,加其他的数据结构来实现是不是就可行呢?因为之前我们就分析过,这个题目本身就是需要通过组合多种数据结构实现的。那么获取随机元素的同时,新增删除又能是O(1)的时间复杂度的,又有谁能?

数组:哥,你看看我啊!这里!

没错,数组,作为最常用并且最先认识的数据结构,有下标的情况下,随机获取是O(1)的复杂度,新增和删除最后一个元素也是O(1)的复杂度。

那么,使用data-index的键值对方式保存到哈希表中,将实际元素放入到数组中。是不是满足新增删除前的判断,并且新增,获取随机元素的时间复杂度都是O(1),唯一没满足的就是删除,因为我们不能保证每次都删除最后一位元素。但是,数组的修改方法,在有下标的时候,时间复杂度是O(1)。正所谓没有条件,创造条件也要上,我们直接把需要删除的元素修改到最后一位,再删除,是不是就变成删除最后一位元素,时间复杂度为O(1)了。

总结一下上面的分析内容:

  1. 哈希表 + 动态数组:使用哈希表来存储元素及其在数组中的位置,使用动态数组来存储实际的元素。
  2. 优化删除操作:为了保证删除操作能在 O(1) 时间内完成,可以将要删除的元素与数组的最后一个元素交换位置,然后删除最后一个元素。

那么,开始表演!

实际算法代码

以下是使用上述思路的 Java 实现:

import java.util.HashMap;
import java.util.Map;class RandomizedSet {private Map<Integer, Integer> valToIndex;private int[] values;private int size;public RandomizedSet() {valToIndex = new HashMap<>();// 初始容量values = new int[8];size = 0;}public boolean insert(int val) {if (valToIndex.containsKey(val)) return false;if (size == values.length) {resizeArray();}valToIndex.put(val, size);values[size] = val;size++;return true;}public boolean remove(int val) {if (!valToIndex.containsKey(val)) return false;int index = valToIndex.get(val);int lastElement = values[size - 1];values[index] = lastElement;valToIndex.put(lastElement, index);valToIndex.remove(val);size--;return true;}public int getRandom() {return values[(int) (Math.random() * size)];}private void resizeArray() {int[] newArray = new int[values.length * 2];System.arraycopy(values, 0, newArray, 0, values.length);values = newArray;}
}

结果

提交到力扣,通过测试:

在这里插入图片描述

总结

最后总结本题中比较重要的两步:

  1. 数据结构设计:使用哈希表和动态数组结合的方式,实现了 O(1) 时间复杂度的插入、删除和获取随机元素操作。
  2. 优化技巧:在删除操作中,通过交换元素位置来避免重新排序,从而保证了 O(1) 的时间复杂度。

其实题目考察的还是对基础数据结构的掌握,掌握的好这个题目难度确实不算很难,但是掌握不好,这种题目有没有暴力破解的解法,就会直接卡死,基础还是重要的啊。

这篇关于力扣面试经典算法150题:O(1) 时间插入、删除和获取随机元素的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1110579

相关文章

Java controller接口出入参时间序列化转换操作方法(两种)

《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

python处理带有时区的日期和时间数据

《python处理带有时区的日期和时间数据》这篇文章主要为大家详细介绍了如何在Python中使用pytz库处理时区信息,包括获取当前UTC时间,转换为特定时区等,有需要的小伙伴可以参考一下... 目录时区基本信息python datetime使用timezonepandas处理时区数据知识延展时区基本信息

SpringBoot整合mybatisPlus实现批量插入并获取ID详解

《SpringBoot整合mybatisPlus实现批量插入并获取ID详解》这篇文章主要为大家详细介绍了SpringBoot如何整合mybatisPlus实现批量插入并获取ID,文中的示例代码讲解详细... 目录【1】saveBATch(一万条数据总耗时:2478ms)【2】集合方式foreach(一万条数

python获取网页表格的多种方法汇总

《python获取网页表格的多种方法汇总》我们在网页上看到很多的表格,如果要获取里面的数据或者转化成其他格式,就需要将表格获取下来并进行整理,在Python中,获取网页表格的方法有多种,下面就跟随小编... 目录1. 使用Pandas的read_html2. 使用BeautifulSoup和pandas3.

SpringBoot UserAgentUtils获取用户浏览器的用法

《SpringBootUserAgentUtils获取用户浏览器的用法》UserAgentUtils是于处理用户代理(User-Agent)字符串的工具类,一般用于解析和处理浏览器、操作系统以及设备... 目录介绍效果图依赖封装客户端工具封装IP工具实体类获取设备信息入库介绍UserAgentUtils

Python的time模块一些常用功能(各种与时间相关的函数)

《Python的time模块一些常用功能(各种与时间相关的函数)》Python的time模块提供了各种与时间相关的函数,包括获取当前时间、处理时间间隔、执行时间测量等,:本文主要介绍Python的... 目录1. 获取当前时间2. 时间格式化3. 延时执行4. 时间戳运算5. 计算代码执行时间6. 转换为指

Java中字符串转时间与时间转字符串的操作详解

《Java中字符串转时间与时间转字符串的操作详解》Java的java.time包提供了强大的日期和时间处理功能,通过DateTimeFormatter可以轻松地在日期时间对象和字符串之间进行转换,下面... 目录一、字符串转时间(一)使用预定义格式(二)自定义格式二、时间转字符串(一)使用预定义格式(二)自

redis过期key的删除策略介绍

《redis过期key的删除策略介绍》:本文主要介绍redis过期key的删除策略,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录第一种策略:被动删除第二种策略:定期删除第三种策略:强制删除关于big key的清理UNLINK命令FLUSHALL/FLUSHDB命

C# foreach 循环中获取索引的实现方式

《C#foreach循环中获取索引的实现方式》:本文主要介绍C#foreach循环中获取索引的实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、手动维护索引变量二、LINQ Select + 元组解构三、扩展方法封装索引四、使用 for 循环替代

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文