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日

相关推荐

  • leetcode191-位1的个数

    原题 编写一个函数,输入是一个无符号整数,返回其二进制表达式中数字位数为 ‘1’ 的个数(也被称为汉明重量)。 示例 1: 输入: 000000000000000000000000…

    算法 2020年4月15日
    0120
  • leetcode104-二叉树的最大深度

    原题 给定一个二叉树,找出其最大深度。 二叉树的深度为根节点到最远叶子节点的最长路径上的节点数。 说明: 叶子节点是指没有子节点的节点。 示例:给定二叉树 [3,9,20,null…

    算法 2020年1月11日
    090
  • leetcode86-分隔链表

    原题 给定一个链表和一个特定值 x,对链表进行分隔,使得所有小于 x 的节点都在大于或等于 x 的节点之前。 你应当保留两个分区中每个节点的初始相对位置。 示例: 输入: head…

    算法 2020年4月27日
    0150
  • leetcode221-最大正方形

    原题 在一个由 0 和 1 组成的二维矩阵内,找到只包含 1 的最大正方形,并返回其面积。 示例: 输入: 1 0 1 0 0 1 0 1 1 1 1 1 1 1 1 1 0 0 …

    算法 2020年5月8日
    0110
  • leetcode1028-从先序遍历还原二叉树

    原题 我们从二叉树的根节点 root 开始进行深度优先搜索。 在遍历中的每个节点处,我们输出 D 条短划线(其中 D 是该节点的深度),然后输出该节点的值。(如果节点的深度为 D,…

    2020年6月18日
    02410
  • leetcode42-接雨水

    原题 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。 上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1] 表示的高…

    2020年1月23日
    0310
  • leetcode3-无重复字符的最长子串

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

    2019年12月26日
    0100
  • 算法竞赛常用数据结构-字典树Trie

    本文参考资源: 详谈树结构(传统树、字典树、hash 树、Merkle Patricia Tree)_C/C++_smilejiasmile的博客-CSDN博客 概述 Tire树称…

    2020年2月17日
    0670
  • leetcode208-实现Trie(前缀树)

    原题 实现一个 Trie (前缀树),包含 insert, search, 和 startsWith 这三个操作。 示例: Trie trie = new Trie(); trie…

    算法 2020年2月22日
    090
  • leetcode70-爬楼梯

    原题 假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢? 注意: 给定 n 是一个正整数。  示例 1: 输入:…

    算法 2020年1月21日
    0850

发表回复

登录后才能评论