insert同时添加两张表_算法总结:左神class6—2哈希表设计RandomPool结构

设计RandomPool结构

【题目】 设计一种结构,在该结构中有如下三个功能:
insert(key):将某个key加入到该结构,做到不重复加入。
delete(key):将原本在结构中的某个key移除。
getRandom():严格等概率随机返回结构中的任何一个key。
【要求】 Insert、delete和getRandom方法的时间复杂度都是 O(1)

【等概率解决思路】

准备两个hashmap和一个变量size,只有add和random的时候,第一个表中用于存放数据,同时记录size大小,第二个表中存放size和对应的数据,对于random返回的时候,则再size中随机抽出一个,然后在第二个表中返回key。(因为 getRandom()函数要求等概率随机返回结构中的任意一个key ,一张哈希表在小样本情况下无法做到严格的等概率,所以需要至少两张哈希表)

【添加】的时候每个在两张表中一一映射,(Key,size)放入Map1,将(size,Key)放入Map2,记录现在的size,即每次执行 insert 操作之后 size 自增

【random】通过math.random来产生index的等概率随机数,再%size,map中的元素做到等概率随机.

1ab062bbe9b9173645ea93bece408a46.png

【删减】的时候,将最后一个值放到被减的桶中。

【解决难点】删除过程中会在原有随机结构中出现好多洞,导致随机性变差   如何解决此问题?  

【答】在Map1中把记录(lastKey,lastIndex) 变为(lastKey,deleteIndex),并在Map2中把记录(deleteIndex, deleteKey)变为(deleteIndex,lastKey)。然后在Map1中删除记录(deleteKey, deleteIndex),并在Map2中把记录(lastIndex,lastKey)删除。最后 size 减 1。这么做相当于把 lastKey 放到了 deleteKey 的位置上,(消除掉map中的洞,保证size中都有东西,保证记录的 index 还是连续的。维持这个连续状态方便后面random,根据当前的 size 随机得到一个 index,保证 index 在范围[0~size-1]上,都对应着有效的 key,然后把 index 对应的 key 返回即可。如果删除的元素给标记之后,map2中不再有这条记录,后面的random需要重新做)。

package com.zuogod.java;import java.util.HashMap;//使用两个哈希表public static class RandomPool {     public HashMap map1 ;     public HashMap map2;     public int size;     public RandomPool() {       map1=  new HashMap<>();       map2= new HashMap<>();       size = 0;       }     public void add(string str){     if (!map1.containsKey(key)){          map1.put(str, size);// 两个表同步加          map2.put(size, str;          size++;     }     public String getRandom(){            if (this.size == 0){                return null;            }            int index = (int)(Math.random() * size); //0 ~ size-1            return map2.get(index);        }    }     public void delete(K key){            if(map1.containsKey(key)){                int deleteIndex = map1.get(key);                int lastIndex = --size;// 拿出最后一个元素                K lastKey = map2.get(lastIndex);                map1.put(lastKey,deleteIndex);                map2.put(deleteIndex,lastKey);                map1.remove(key);                map2.remove(lastIndex);            }        })    public static void main(String[] args) {        Pool pool = new Pool();        pool.insert("zuo");        pool.insert("cheng");        pool.insert("yun");        pool.delete("zuo");        System.out.println(pool.getRandom());        System.out.println(pool.getRandom());        System.out.println(pool.getRandom());        System.out.println(pool.getRandom());        System.out.println(pool.getRandom());    }}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值