zoukankan      html  css  js  c++  java
  • 438. Find All Anagrams in a String

    Question

    438. Find All Anagrams in a String

    Solution

    题目大意:给两个字符串,s和p,求p在s中出现的位置,p串中的字符无序,ab=ba

    思路:起初想的是求p的全排列,保存到set中,遍历s,如果在set中出现,s中的第一个字符位置保存到结果中,最后返回结果。这种思路执行超时。可能是求全排列超时的。

    思路2:先把p中的字符及字符出现的次数统计出来保存到map中,再遍历s,这个思路和169. Majority Element - LeetCode中提到的创新解法类似

    Java实现:

    public List<Integer> findAnagrams(String s, String p) {
        List<Integer> ans = new ArrayList<>();
        if (s.length() <= p.length()) return ans;
    
        // 构造map,并初始化target
        Map<Character, Bo> map = new HashMap<>();
        for (char tmp : p.toCharArray()) {
            Bo bo = map.get(tmp);
            if (bo == null) {
                bo = new Bo();
                map.put(tmp, bo);
            }
            bo.target++;
        }
    
        // 前p.length()项
        for (int i = 0; i < p.length(); i++) {
            char cur = s.charAt(i);
            Bo bo = map.get(cur);
            if (bo != null) {
                bo.cur++;
            }
        }
        if (allOne(map)) ans.add(0);
    
        for (int i = p.length(); i < s.length(); i++) {
            char cur = s.charAt(i);
            if (map.get(cur) != null) map.get(cur).cur++;
            char last = s.charAt(i - p.length());
            if (map.get(last) != null) map.get(last).cur--;
            if (allOne(map)) ans.add(i - p.length() + 1);
        }
    
        return ans;
    }
    
    public boolean allOne(Map<Character, Bo> map) {
        for (Map.Entry<Character, Bo> entry : map.entrySet()) {
            if (entry.getValue().cur != entry.getValue().target) return false;
        }
        return true;
    }
    
    class Bo {
        int cur; // 当前数
        int target; // 目标数
    
        public Bo() {
            this(0, 0);
        }
    
        public Bo(int cur, int target) {
            this.cur = cur;
            this.target = target;
        }
    }
    

    Discuss

    欣赏一下别人写的,所说下面两道题用的是同一思路,记录一下

    https://leetcode.com/problems/find-all-anagrams-in-a-string/discuss/92015/ShortestConcise-JAVA-O(n)-Sliding-Window-Solution

    https://leetcode.com/problems/minimum-window-substring/discuss/26808/here-is-a-10-line-template-that-can-solve-most-substring-problems

    public List<Integer> findAnagrams(String s, String p) {
        List<Integer> list = new ArrayList<>();
        if (s == null || s.length() == 0 || p == null || p.length() == 0) return list;
        int[] hash = new int[256]; //character hash
        //record each character in p to hash
        for (char c : p.toCharArray()) {
            hash[c]++;
        }
        //two points, initialize count to p's length
        int left = 0, right = 0, count = p.length();
        while (right < s.length()) {
            //move right everytime, if the character exists in p's hash, decrease the count
            //current hash value >= 1 means the character is existing in p
            if (hash[s.charAt(right++)]-- >= 1) count--; 
            
            //when the count is down to 0, means we found the right anagram
            //then add window's left to result list
            if (count == 0) list.add(left);
        
            //if we find the window's size equals to p, then we have to move left (narrow the window) to find the new match window
            //++ to reset the hash because we kicked out the left
            //only increase the count if the character is in p
            //the count >= 0 indicate it was original in the hash, cuz it won't go below 0
            if (right - left == p.length() && hash[s.charAt(left++)]++ >= 0) count++;
        }
        return list;
    }
    
    public List<Integer> findAnagrams(String s, String p) {
    
        char[] ptrn = p.toCharArray();
        char[] str = s.toCharArray();
        
        int[] w = new int[26];
        
        for(char c : ptrn) w[c - 'a']++;
        
        int start = 0;
                
        List<Integer> result = new LinkedList<>();
        
        for(int i = 0; i<str.length; i++){
            int cIndex = str[i] - 'a';
    
            w[cIndex]--;
            // the crucial bit, if we have seen the character too many times 
            // or it is a character that is not in the pattern, rewind the starting index
            while(w[cIndex] < 0){
                w[str[start] - 'a']++;
                start++;
            }
            
            if(i - start + 1 == ptrn.length){
                result.add(start);
                w[str[start] - 'a']++;
                start++;
            }
        }
        
    
        return result;
    }      
    
  • 相关阅读:
    【图像处理】【SEED-VPM】7.ubuntu10.04下 TFTP,NFS 安装指南
    【图像处理】【SEED-VPM】6.文件目录结构
    【DIY】【外壳】木板 & 亚克力 加工
    【图像处理】【SEED-VPM】5.uImage的烧写 & NFS烧写文件系统
    【PCB】【AD使用】Altium Designer 的entry sheet ,offsheet和port作用
    【PCB】【项目记录】AWG任意波形产生器
    【图像处理】【SEED-VPM】4.串口调试信息
    【图像处理】【SEED-VPM】3.外设信息
    【图像处理】【SEED-VPM】2.接口
    【图像处理】【SEED-VPM】1.注意点
  • 原文地址:https://www.cnblogs.com/okokabcd/p/9324553.html
Copyright © 2011-2022 走看看