[LeetCode] 981. Time Based Key-Value Store

Design a time-based key-value data structure that can store multiple values for the same key at different time stamps and retrieve the key's value at a certain timestamp.

Implement the TimeMap class:

  • TimeMap() Initializes the object of the data structure.
  • void set(String key, String value, int timestamp) Stores the key key with the value value at the given time timestamp.
  • String get(String key, int timestamp) Returns a value such that set was called previously, with timestamp_prev <= timestamp. If there are multiple such values, it returns the value associated with the largest timestamp_prev. If there are no values, it returns "".

Example 1:
Input
["TimeMap", "set", "get", "get", "set", "get", "get"]
[[], ["foo", "bar", 1], ["foo", 1], ["foo", 3], ["foo", "bar2", 4], ["foo", 4], ["foo", 5]]
Output
[null, null, "bar", "bar", null, "bar2", "bar2"]

Explanation
TimeMap timeMap = new TimeMap();
timeMap.set("foo", "bar", 1); // store the key "foo" and value "bar" along with timestamp = 1.
timeMap.get("foo", 1); // return "bar"
timeMap.get("foo", 3); // return "bar", since there is no value corresponding to foo at timestamp 3 and timestamp 2, then the only value is at timestamp 1 is "bar".
timeMap.set("foo", "bar2", 4); // store the key "foo" and value "bar2" along with timestamp = 4.
timeMap.get("foo", 4); // return "bar2"
timeMap.get("foo", 5); // return "bar2"

Constraints:
1 <= key.length, value.length <= 100
key and value consist of lowercase English letters and digits.
1 <= timestamp <= 107
All the timestamps timestamp of set are strictly increasing.
At most 2 * 105 calls will be made to set and get.

基于时间的键值存储。

设计一个基于时间的键值数据结构,该结构可以在不同时间戳存储对应同一个键的多个值,并针对特定时间戳检索键对应的值。

实现 TimeMap 类:

TimeMap() 初始化数据结构对象
void set(String key, String value, int timestamp) 存储键 key、值 value,以及给定的时间戳 timestamp。
String get(String key, int timestamp)
返回先前调用 set(key, value, timestamp_prev) 所存储的值,其中 timestamp_prev <= timestamp 。
如果有多个这样的值,则返回对应最大的 timestamp_prev 的那个值。
如果没有值,则返回空字符串("")。

来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/time-based-key-value-store
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路一 - treemap

这个题有两种思路,一是用Java的 treemap,另一种是用到 binary search。

首先第一种思路,用 treemap。因为会存在多个键值是同样的 key,但是有不同的 value 和不同的 timestamp,所以一开始比较直观的思路一定是 hashmap<String, timestamp>,因为 hashmap 才能存储类似这种<key, value>键值对的数据。其次,题目要求当找到同样的 key,同样的 value 的时候,要返回最接近 timestamp 的 value,所以想到用 treemap<timestamp, String>,因为 treemap 是可以自动根据 key 排序的。

复杂度

  • treemap
    • 构造 treemap 的时间O(nlogn) - 每放进去一个元素就要重新排序一次
    • 空间O(n)

代码

treemap实现

class TimeMap {
	private HashMap<String, TreeMap<Integer, String>> map;

	/** Initialize your data structure here. */
	public TimeMap() {
		map = new HashMap<>();
	}

	public void set(String key, String value, int timestamp) {
		TreeMap<Integer, String> tmap = map.get(key);
		if (tmap == null) {
			tmap = new TreeMap<>();
			map.put(key, tmap);
		}
		tmap.put(timestamp, value);
	}

	public String get(String key, int timestamp) {
		TreeMap<Integer, String> tmap = map.get(key);
		if (tmap == null) {
			return "";
		}
		Integer floorKey = tmap.floorKey(timestamp);
		if (floorKey == null) {
			return "";
		}
		return tmap.get(floorKey);
	}
}

/**
 * Your TimeMap object will be instantiated and called as such:
 * TimeMap obj = new TimeMap();
 * obj.set(key,value,timestamp);
 * String param_2 = obj.get(key,timestamp);
 */

思路二 - 二分法

第二种思路是用到二分法。在不用 treemap 的前提下,我们依然需要 hashmap 存储 key 和 value(这里会是一个 list)的键值对,但是对于 value 相同但是 timestamp 不同的数据,我们可以自己创建一个类,把这个类放入 list,这样当 get key 的时候,实际上我们 get 到的是一个由 arraylist 串联好的,有相同 key 的 values。当你能得到这个 list 之后,可以根据给出的变量 timestamp 找到 list 中最接近于这个 timestamp 的键值对。因为题目给的时间戳一定是有序的所以这里我们无需再排序了。

复杂度

  • binary search
    • 时间O(logn)
    • 空间O(n)

代码

binary search实现

class Data {
    String val;
    int time;
    
    public Data(String val, int time) {
        this.val = val;
        this.time = time;
    }
}

class TimeMap {
    HashMap<String, List<Data>> map;

    public TimeMap() {
        map = new HashMap<>();
    }
    
    public void set(String key, String value, int timestamp) {
        if (!map.containsKey(key)) {
            map.put(key, new ArrayList<>());
        }
        map.get(key).add(new Data(value, timestamp));
    }
    
    public String get(String key, int timestamp) {
        if (!map.containsKey(key)) {
            return "";
        }
        return helper(map.get(key), timestamp);
    }

    private String helper(List<Data> list, int timestamp) {
        int left = 0;
        int right = list.size() - 1;
        while (left + 1 < right) {
            int mid = left + (right - left) / 2;
            if (list.get(mid).time == timestamp) {
                return list.get(mid).val;
            } else if (list.get(mid).time < timestamp) {
                left = mid;
            } else if (list.get(mid).time > timestamp) {
                right = mid;
            }
        }
		// 因为如果right <= timestamp,那么right一定是更接近timestamp的时间
        if (list.get(right).time <= timestamp) {
            return list.get(right).val;
        }
        if (list.get(left).time <= timestamp) {
            return list.get(left).val;
        }
        return "";
    }
}

/**
 * Your TimeMap object will be instantiated and called as such:
 * TimeMap obj = new TimeMap();
 * obj.set(key,value,timestamp);
 * String param_2 = obj.get(key,timestamp);
 */

相关题目

981. Time Based Key-Value Store
2034. Stock Price Fluctuation
posted @ 2020-05-16 04:30  CNoodle  阅读(263)  评论(0编辑  收藏  举报