leetcode380-常数时间插入、删除和获取随机元素

原题

设计一个支持在平均 时间复杂度 O(1) 下,执行以下操作的数据结构。

  1. insert(val):当元素 val 不存在时,向集合中插入该项。
  2. remove(val):元素 val 存在时,从集合中移除该项。
  3. getRandom:随机返回现有集合中的一项。每个元素应该有相同的概率被返回。

示例:

// 初始化一个空的集合。
RandomizedSet randomSet = new RandomizedSet();

// 向集合中插入 1 。返回 true 表示 1 被成功地插入。
randomSet.insert(1);

// 返回 false ,表示集合中不存在 2 。
randomSet.remove(2);

// 向集合中插入 2 。返回 true 。集合现在包含 [1,2] 。
randomSet.insert(2);

// getRandom 应随机返回 1 或 2 。
randomSet.getRandom();

// 从集合中移除 1 ,返回 true 。集合现在包含 [2] 。
randomSet.remove(1);

// 2 已在集合中,所以返回 false 。
randomSet.insert(2);

// 由于 2 是集合中唯一的数字,getRandom 总是返回 2 。
randomSet.getRandom();

解法

思想

  1. 初始想法

哈希表的插入删除的时间复杂度都是O(1),获取的时候可以通过EntrySet。
所以这道题是不能用HashSet的。

这样虽然获取随机元素的时候时间复杂度最高可能是O(n),但仍比遍历一遍Set转ArrayList好很多。

  1. 正确解法

哈希表插入和删除都是O(1),而顺序表随机访问则是O(1),可以使用ArrayList来存储所有的数据。但是必须解决ArrayList删除元素的O(n)问题。

于是可以:

在哈希表中用value-index来记录值和在list中的下标的对应关系,如图所示

leetcode380-常数时间插入、删除和获取随机元素

当删除元素时,size减一,用list中最后那个元素替换要删除的那个元素,并且将哈希表中的对应关系改过来(用要删除的元素的index替换list中最后那个元素对应的index):

leetcode380-常数时间插入、删除和获取随机元素

此时若要随机访问元素,只需获取list中前3(size)个元素中的一个。

那么如果需要继续插入元素,只需从list中下标为3(size)处替换掉后面那个元素或是在后面那个元素之前插入(这里如果用插入是使用add(index,value)方法,个人觉得比起set会增加时间复杂度,因为使用add后面的元素都需要向后移动,虽然jdk源码中使用了System.arraycopy即内存拷贝来优化,但是也比直接替换的时间复杂度更高):

leetcode380-常数时间插入、删除和获取随机元素

如果不是替换元素而是add操作,这里会变成1-4-2-8-4

代码

  1. 初始想法
class RandomizedSet {
    Object none = new Object();
    HashMap<Integer,Object> map;
    Random random;
    /** Initialize your data structure here. */
    public RandomizedSet() {
        map = new HashMap<>();
        random = new Random();
    }

    /** Inserts a value to the set. Returns true if the set did not already contain the specified element. */
    public boolean insert(int val) {
        if(map.containsKey(val)) return false;
        map.put(val,none);
        return true;
    }

    /** Removes a value from the set. Returns true if the set contained the specified element. */
    public boolean remove(int val) {
        if(!map.containsKey(val)) return false;
        map.remove(val);
        return true;
    }

    /** Get a random element from the set. */
    public int getRandom() {
        int ran = random.nextInt(map.size());  
        int n = 0;
        for(Map.Entry<Integer,Object> i:map.entrySet()){
            if(n==ran) return i.getKey();
            n++;
        }
        return 0;
    }
}

/**
 * Your RandomizedSet object will be instantiated and called as such:
 * RandomizedSet obj = new RandomizedSet();
 * boolean param_1 = obj.insert(val);
 * boolean param_2 = obj.remove(val);
 * int param_3 = obj.getRandom();
 */
  1. 正确解法(来源:leetcode用户,添加元素时是直接插入)
class RandomizedSet {

    Map<Integer,Integer> map; // 存放值和在 list 的下标位置的映射
    List<Integer> list;       // 存放要插入数据的结构
    int size;                 // 数据的长度
    /** Initialize your data structure here. */
    public RandomizedSet() {
        map = new HashMap<>();
        list = new ArrayList<>();
        size = 0;
    }

    /** Inserts a value to the set. Returns true if the set did not already contain the specified element. */
    public boolean insert(int val) {
        if(map.containsKey(val)) return false;
        else{
            // 插入数据,并更新 map 的映射后将长度加一
            list.add(size,val);
            map.put(val,size++);
            return true;
        }
    }

    /** Removes a value from the set. Returns true if the set contained the specified element. */
    public boolean remove(int val) {
        if(!map.containsKey(val)) return false;
        else if( size == 0 ){ map.remove(val);}
        else{
            // 取到 list 末尾的数据
            int tailKey = list.get(size-1);
            // 然后将要原先 map 中得 val-index 映射改为 tailKey-index
            map.put(tailKey,map.get(val));
            // 在 map 中取得 val 在 list 的位置,然后根据这个位置用末尾元素 tailKey 替代
            list.set(map.get(val),tailKey);
            // 在 map 中删除 val 的映射
            map.remove(val);
            size--;
        }
        return true;
    }

    /** Get a random element from the set. */
    public int getRandom() {
        Random rand = new Random();
         // rand.nextInt(size) 产生的是 0 到 size(不包括 size) 的数据
        return list.get(rand.nextInt(size));
    }
}
type RandomizedSet struct {
    Value2Index map[int]int
    List []int
}


func Constructor() RandomizedSet {
    return RandomizedSet{
        Value2Index: map[int]int{},
        List: []int{},
    }
}


func (this *RandomizedSet) Insert(val int) bool {
    _, ok := this.Value2Index[val] 
    if ok {
        return false
    }
    this.List = append(this.List, val)
    this.Value2Index[val] = len(this.List) - 1
    return true
}


func (this *RandomizedSet) Remove(val int) bool {
    index, ok := this.Value2Index[val] 
    if !ok {
        return false
    }
    delete(this.Value2Index, val)

    length := len(this.List)
    if length > 0{
        // 把最后一个元素填补到要删除的元素位置上
        lastPosValue := this.List[length - 1]
        this.List = this.List[0:length - 1]
        // 如果要删的元素本身就是最后一个元素,清空数组即可
        if index != length - 1{
            this.List[index] = lastPosValue
            this.Value2Index[lastPosValue] = index
        }
    } 
    return true
}


func (this *RandomizedSet) GetRandom() int {
    length := len(this.List)
    if length == 0{
        return 0
    }
    return this.List[rand.Intn(length)]
}

原创文章,作者:彭晨涛,如若转载,请注明出处:https://www.codetool.top/article/leetcode380-%e5%b8%b8%e6%95%b0%e6%97%b6%e9%97%b4%e6%8f%92%e5%85%a5%e3%80%81%e5%88%a0%e9%99%a4%e5%92%8c%e8%8e%b7%e5%8f%96%e9%9a%8f%e6%9c%ba%e5%85%83%e7%b4%a0/

(0)
彭晨涛彭晨涛管理者
上一篇 2019年12月28日
下一篇 2019年12月29日

相关推荐

  • leetcode224-基本计算器

    原题 实现一个基本的计算器来计算一个简单的字符串表达式的值。 字符串表达式可以包含左括号 ( ,右括号 ),加号 + ,减号 -,非负整数和空格 。 示例1: 输入: "1 + 1…

    算法 2020年1月26日
    0160
  • leetcode3-无重复字符的最长子串

    原题 给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度。 示例1: 输入: "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是 "abc",所以其长…

    2019年12月26日
    0100
  • leetcode74-搜索二维矩阵

    原题 编写一个高效的算法来判断 m x n 矩阵中,是否存在一个目标值。该矩阵具有如下特性: 每行中的整数从左到右按升序排列。 每行的第一个整数大于前一行的最后一个整数。   示例…

    算法 2020年4月29日
    0210
  • leetcode153-寻找旋转排序数组中的最小值

    原题 假设按照升序排序的数组在预先未知的某个点上进行了旋转。 ( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] )。 搜索一个给定的目标值,…

    算法 2020年1月3日
    080
  • 程序员面试金典08.01-三步问题

    原题(来源Leetcode) 三步问题。有个小孩正在上楼梯,楼梯有n阶台阶,小孩一次可以上1阶、2阶或3阶。实现一种方法,计算小孩有多少种上楼梯的方式。结果可能很大,你需要对结果模…

    算法 2020年6月19日
    04800
  • leetcode13-罗马数字转整数

    原题 罗马数字包含以下七种字符: I, V, X, L,C,D 和 M。 字符 数值 I 1 V 5 X 10 L 50 C 100 D 500 M 1000 例如, 罗马数字 2…

    算法 2020年4月25日
    0100
  • leetcode134-加油站

    原题 在一条环路上有 N 个加油站,其中第 i 个加油站有汽油 gas[i] 升。 你有一辆油箱容量无限的的汽车,从第 i 个加油站开往第 i+1 个加油站需要消耗汽油 cost[…

    算法 2020年2月16日
    0100
  • leetcode350-两个数组的交集II

    原题 给定两个数组,编写一个函数来计算它们的交集。 输入: nums1 = [1,2,2,1], nums2 = [2,2] 输出: [2,2] 示例2: 输入: nums1 = …

    算法 2019年12月23日
    0510
  • leetcode297-二叉树的序列化与反序列化

    原题 序列化是将一个数据结构或者对象转换为连续的比特位的操作,进而可以将转换后的数据存储在一个文件或者内存中,同时也可以通过网络传输到另一个计算机环境,采取相反方式重构得到原数据。…

    算法 2020年1月15日
    0120
  • 海量数据去重-由BitMap引出的布隆过滤器

    本文参考资源: 那些惊艳的算法们(一)——布隆过滤器_C/C++_xinzhongtianxia的博客-CSDN博客 详解布隆过滤器的原理、使用场景和注意事项 - 简书 概述 昨天…

    2020年2月28日
    01.4K0

发表回复

登录后才能评论