432. 全 O(1) 的数据结构

2019-04-27  本文已影响0人  薄荷糖的味道_fb40

实现一个数据结构支持以下操作:

Inc(key) - 插入一个新的值为 1 的 key。或者使一个存在的 key 增加一,保证 key 不为空字符串。
Dec(key) - 如果这个 key 的值是 1,那么把他从数据结构中移除掉。否者使一个存在的 key 值减一。如果这个 key 不存在,这个函数不做任何事情。key 保证不为空字符串。
GetMaxKey() - 返回 key 中值最大的任意一个。如果没有元素存在,返回一个空字符串""。
GetMinKey() - 返回 key 中值最小的任意一个。如果没有元素存在,返回一个空字符串""。
挑战:以 O(1) 的时间复杂度实现所有操作。

思路:

这好像是桶排序吧。
1.以出现次数为key创建桶链表,每个桶内装有所有出现次数为某个值的变量,每个key代表一个桶。
2.以输入的string为key创建哈希表,value为对应的桶。
实现代码如下。

class AllOne {
public:
    /** Initialize your data structure here. */
    AllOne() {
        
    }
    
    /** Inserts a new key <Key> with value 1. Or increments an existing key by 1. */
    void inc(string key) {
        
        if(!hashmap.count(key))
        {
            if(buckets.empty() || buckets.back().val!=1)
            {
                auto currBucket=buckets.insert(buckets.end(),{1,{key}});
                hashmap[key]=currBucket;
            }
            else
            {
                auto currBucket=--buckets.end();
                currBucket->keys.insert(key);
                hashmap[key]=currBucket;
            }
        }
        else
        {
            auto currBucket=hashmap[key];
            if(currBucket==buckets.begin())
            {
                auto newBucket=buckets.insert(currBucket,{currBucket->val+1,{key}});
                hashmap[key]=newBucket;
            }
            else
            {
                auto lastBucket=--hashmap[key];
                if(lastBucket->val!=currBucket->val+1)
                {
                    auto newBucket=buckets.insert(currBucket,{currBucket->val+1,{key}});
                    hashmap[key]=newBucket;
                }
                else
                {
                    lastBucket->keys.insert(key);
                    hashmap[key]=lastBucket;
                }
                
                
            }
            currBucket->keys.erase(key);
            if(currBucket->keys.empty())
            {
                buckets.erase(currBucket);
            }
        }
    }
    
    /** Decrements an existing key by 1. If Key's value is 1, remove it from the data structure. */
    void dec(string key) {
        
        if(hashmap.count(key))
        {
            auto currBucket=hashmap[key];
            if(currBucket->val==1)
            {
                currBucket->keys.erase(key);
                if(currBucket->keys.empty())
                {
                    buckets.erase(currBucket);
                }
                hashmap.erase(key);
                return;
            }
            else
            {
                auto nextBucket=++hashmap[key];
                if(nextBucket==buckets.end() || nextBucket->val!=currBucket->val-1)
                {
                    auto newBucket=buckets.insert(nextBucket,{currBucket->val-1,{key}});
                    hashmap[key]=newBucket;
                    currBucket->keys.erase(key);
                }
                else
                {
                    nextBucket->keys.insert(key);
                    hashmap[key]=nextBucket;
                    currBucket->keys.erase(key);
                }
            }
            if(currBucket->keys.empty())
            {
                buckets.erase(currBucket);
            }
        }
    }
    
    /** Returns one of the keys with maximal value. */
    string getMaxKey() {
        return buckets.empty()?"":*(buckets.begin()->keys.begin());
    }
    
    /** Returns one of the keys with Minimal value. */
    string getMinKey() {
        return buckets.empty()?"":*(buckets.rbegin()->keys.begin());
    }
private:
    struct bucket{
        int val;
        unordered_set<string> keys;
    };
    list<bucket> buckets;
    unordered_map<string,list<bucket>::iterator> hashmap;
};

/**
 * Your AllOne object will be instantiated and called as such:
 * AllOne* obj = new AllOne();
 * obj->inc(key);
 * obj->dec(key);
 * string param_3 = obj->getMaxKey();
 * string param_4 = obj->getMinKey();
 */
上一篇 下一篇

猜你喜欢

热点阅读