力扣面试经典算法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

相关文章

使用C#代码在PDF文档中添加、删除和替换图片

《使用C#代码在PDF文档中添加、删除和替换图片》在当今数字化文档处理场景中,动态操作PDF文档中的图像已成为企业级应用开发的核心需求之一,本文将介绍如何在.NET平台使用C#代码在PDF文档中添加、... 目录引言用C#添加图片到PDF文档用C#删除PDF文档中的图片用C#替换PDF文档中的图片引言在当

macOS无效Launchpad图标轻松删除的4 种实用方法

《macOS无效Launchpad图标轻松删除的4种实用方法》mac中不在appstore上下载的应用经常在删除后它的图标还残留在launchpad中,并且长按图标也不会出现删除符号,下面解决这个问... 在 MACOS 上,Launchpad(也就是「启动台」)是一个便捷的 App 启动工具。但有时候,应

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Mysql删除几亿条数据表中的部分数据的方法实现

《Mysql删除几亿条数据表中的部分数据的方法实现》在MySQL中删除一个大表中的数据时,需要特别注意操作的性能和对系统的影响,本文主要介绍了Mysql删除几亿条数据表中的部分数据的方法实现,具有一定... 目录1、需求2、方案1. 使用 DELETE 语句分批删除2. 使用 INPLACE ALTER T

MySQL INSERT语句实现当记录不存在时插入的几种方法

《MySQLINSERT语句实现当记录不存在时插入的几种方法》MySQL的INSERT语句是用于向数据库表中插入新记录的关键命令,下面:本文主要介绍MySQLINSERT语句实现当记录不存在时... 目录使用 INSERT IGNORE使用 ON DUPLICATE KEY UPDATE使用 REPLACE

使用Python实现获取网页指定内容

《使用Python实现获取网页指定内容》在当今互联网时代,网页数据抓取是一项非常重要的技能,本文将带你从零开始学习如何使用Python获取网页中的指定内容,希望对大家有所帮助... 目录引言1. 网页抓取的基本概念2. python中的网页抓取库3. 安装必要的库4. 发送HTTP请求并获取网页内容5. 解

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque