2020.7.31

146. LRU缓存机制

运用你所掌握的数据结构,设计和实现一个 LRU (最近最少使用) 缓存机制。它应该支持以下操作: 获取数据 get 和 写入数据 put 。
获取数据 get(key) - 如果关键字 (key) 存在于缓存中,则获取关键字的值(总是正数),否则返回 -1。
写入数据 put(key, value) - 如果关键字已经存在,则变更其数据值;如果关键字不存在,则插入该组「关键字/值」。当缓存容量达到上限时,它应该在写入新数据之前删除最久未使用的数据值,从而为新的数据值留出空间。
进阶:
你是否可以在 O(1) 时间复杂度内完成这两种操作?

JAVA实现

import java.util.HashMap;

public class LRUCache {
    HashMap<Integer, Node> map;
    private int capacity;
    // 虚拟头尾结点
    private Node first;
    private Node last;

    public LRUCache(int capacity){
        map = new HashMap<>(capacity);
        first = new Node();// Node内部创建一个空的初始化方法
        last = new Node();
        first.next = last;
        last.pre = first;
        this.capacity = capacity;
    }

    public int get(int key) {
         Node node = map.get(key);
        
         if(node != null) {
             removeNode(node);
             addAfterFirst(node);
         }
        return  (node != null) ? node.value : -1;
    }

    private void removeNode(Node node) {
        node.pre.next = node.next;
        node.next.pre = node.pre;
    }

    private void addAfterFirst(Node node) {
        // node 与first.next 关系
        node.next = first.next;
        first.next.pre = node;

        // first 与 node 关系
        first.next = node;
        node.pre = first;

    }
    
    public  void put(int key, int value) {
        Node node = map.get(key);
        if (node != null) { // 新值覆盖旧值
            node.value = value;
            removeNode(node);
            addAfterFirst(node);
        }else {// 添加一对新的key value
            if (map.size() == capacity) {
                // 两件事情,第一件把key从map上删除, 第二件事情节点从双向链表中删掉
                map.remove(last.pre.key);
                removeNode(last.pre);
            }

            Node newNode = new  Node(key, value);
            map.put(key, newNode);
            addAfterFirst(newNode);
        }
    }

     private static class Node {
        public int key;
        public int value;
        public Node pre;
        public Node next;

        public Node(int key, int value) {
            this.key = key;
            this.value = value;
        }
        public Node(){} // 保留空的构造方法给first和next使用
    }
}

56 合并区间

给出一个区间的集合,请合并所有重叠的区间。
示例 1:
输入: [[1,3],[2,6],[8,10],[15,18]]
输出: [[1,6],[8,10],[15,18]]
解释: 区间 [1,3] 和 [2,6] 重叠, 将它们合并为 [1,6].
示例 2:
输入: [[1,4],[4,5]]
输出: [[1,5]]
解释: 区间 [1,4] 和 [4,5] 可被视为重叠区间。

class Solution:
    def merge(self, intervals: List[List[int]]) -> List[List[int]]:
        if not intervals: return []
	n = len(intervals)
	intervals.sort()
	res =[]
	left = intervals[0][0]
	right = intervals[0][1]
	for i in range(1, n):
		if(intervals[i][0] <= right):
			if(intervals[i][1] > right):
				right=intervals[i][1]
		else:
			res.append([left, right])
			left = intervals[i][0]
			right = intervals[i][1]
      res.append([left, right])
      return res
class Solution:
    def merge(self, intervals: List[List[int]]) -> List[List[int]]:
        if not intervals: return []
	n = len(intervals)
	intervals = sorted(intervals)
	res =[]
	i=0
	while(i<n):
		left = intervals[i][0]
		right = intervals[i][1]
		while(i<n-1 and intervals[i+1][0]<=right):
			i+=1
			right=max(intervals[i][1],right)
		res.append([left, right])
		i+=1
      return res
class Solution:
    def merge(self, intervals: List[List[int]]) -> List[List[int]]:
        if not intervals: return []
        intervals.sort()
        res = [intervals[0]]
        for x, y in intervals[1:]:
            if res[-1][1] < x:
                res.append([x, y])
            else:
                res[-1][1] = max(y, res[-1][1])
        return res

242. 有效的字母异位词

给定两个字符串 s 和 t ,编写一个函数来判断 t 是否是 s 的一个字母异位词。
示例 1:
输入: s = “anagram”, t = “nagaram”
输出: true
示例 2:
输入: s = “rat”, t = “car”
输出: false
说明:
你可以假设字符串只包含小写字母。
进阶:
如果输入字符串包含 unicode 字符怎么办?你能否调整你的解法来应对这种情况?
思路1
用map的key存该字符,value存该字符出现的次数,再用该map去检测字符串t,有这个字符,相对应的value减一

class Solution {
    public boolean isAnagram(String s, String t) {
       HashMap<Character,Integer> map=new HashMap<Character,Integer>();
        char s1[]=s.toCharArray();
        char t1[]=t.toCharArray();
        if(s1.length==0&&t1.length==0){
            return true;
        }
        for(int i=0;i<s1.length;++i){
            if(map.containsKey(s1[i])){
                map.put(s1[i],map.get(s1[i])+1);
            }else{
                map.put(s1[i],1);
            }
            
        }
        
        for(int i=0;i<t1.length;++i){
            if(map.containsKey(t1[i])){
                map.put(t1[i],map.get(t1[i])-1);
            }else{
                return false;
            }
        }
        int c=0;
        for(int i=0;i<s1.length;++i){
            if(map.get(s1[i])==0){
                ++c;
                if(c==s1.length){
                    return true;
                }
            }else{
                return false;
            }
        }
        return false;
    }
}

思路2
用s[i]-'a’作为sflag的下标,值就是该字符出现的次数,对两个字符串都进行此操作,最后对比次数,完全一样就返回true

bool isAnagram(char * s, char * t){
     int len1 = strlen(s);
    int len2 = strlen(t);
    int sflag[26] = {0};
    int tflag[26] = {0};
    
    for (int i = 0; i < len1; i++) {
        sflag[s[i] - 'a'] += 1;
    }
    
    for (int i = 0; i < len2; i++) {
        tflag[t[i] - 'a'] += 1;
    }
    
    for(int i = 0; i < 26; ++i) {
        if (sflag[i] != tflag[i]) {
            return false;
        }
    }
    return true;
}

你可能感兴趣的:(日程执行计划和实现结果)