[CC150v5] 17.14 Optimal Way to Unconcatenate Doc


[CC150v5] 17.14 Optimal Way to Unconcatenate Doc - Shuatiblog.com
https://www.cnblogs.com/grandyang/p/5448539.html
17.14 Oh, no! You have just completed a lengthy document when you have an unfortunate Find/Replace mishap. You have accidentally removed all spaces, punctuation, and capitalization in the document. A sentence like "I reset the computer. It still didn't boot!" would become "iresetthecomputeritstilldidntboot". You figure that you can add back in the punctation and capitalization later, once you get the individual words properly separated. Most of the words will be in a dictionary, but some strings, like proper names, will not.
Given a dictionary (a list of words), design an algorithm to find the optimal way of "unconcatenating" a sequence of words. In this case, "optimal" is defined to be the parsing which minimizes the number of unrecognized sequences of characters.
For example, the string "jesslookedjustliketimherbrother" would be optimally parsed as "JESS looked just like TIM her brother". This parsing has seven unrecognized characters, which we have capitalized for clarity

https://www.cnblogs.com/EdwardLiu/p/4356967.html
这是CareerCup Chapter 17的第14题,我没怎么看CareerCup上的解法,但感觉这道题跟Word BreakPalindrome Partition II很像,都是有一个dictionary, 可以用一维DP来做,用一个int[] res = new int[len+1]; res[i] refers to minimized # of unrecognized chars in first i chars, res[0]=0, res[len]即为所求。
有了维护量,现在需要考虑转移方程,如下:
int unrecogNum = dict.contains(s.substring(j, i))? 0 : i-j; //看index从j到i-1的substring在不在dictionary里,如果不在,unrecogNum=j到i-1的char数
res[i] = Math.min(res[i], res[j]+unrecogNum);
 8     public int optway(String s, Set<String> dict) {
 9         if (s==null || s.length()==0) return 0;
10         int len = s.length();
11         if (dict.isEmpty()) return len;
12         int[] res = new int[len+1]; // res[i] refers to minimized # of unrecognized chars in first i chars
13         Arrays.fill(res, Integer.MAX_VALUE);
14         res[0] = 0;
15         for (int i=1; i<=len; i++) {
16             for (int j=0; j<i; j++) {
17                 String str = s.substring(j, i);
18                 int unrecogNum = dict.contains(str)? 0 : i-j;
19                 res[i] = Math.min(res[i], res[j]+unrecogNum);
20             }
21         }
22         return res[len];
23     }

X.Trie
    private static String reSpace(String input, String... dict) {
        Trie trie = new Trie(dict);
        char[] chars = input.toCharArray();
        Map<Character, Letter> letters = trie.letters;
        Letter lastLetter = null;
        StringBuilder sb = new StringBuilder();
        int unrecognizedCharsCount = 0;
        for (int i = 0; i < chars.length; i++) {
            int j = 0;
            while (i + j < chars.length && letters.containsKey(chars[i + j])) {
                lastLetter = letters.get(chars[i + j]);
                letters = lastLetter.children;
                j++;
            }

            if (lastLetter != null && lastLetter.wordEnd) {
                if (sb.length() > 0 && sb.charAt(sb.length() - 1) != ' ') {
                    sb.append(' ');
                }
                sb.append(new String(Arrays.copyOfRange(chars, i, i + j)));
                sb.append(' ');
                i += (j - 1);
            } else {
                sb.append(chars[i]);
                unrecognizedCharsCount++;
            }
            lastLetter = null;
            letters = trie.letters;
        }

        System.out.println("Out: " + sb);
        System.out.println("Unrecognized chars count: " + unrecognizedCharsCount);
        return sb.toString();
    }

    private static class Trie {
        private Map<Character, Letter> letters = new HashMap<>();

        public Trie(String... strings) {
            for (String string : strings) {
                Map<Character, Letter> candidates = letters;
                char[] charArray = string.toCharArray();
                for (int i = 0; i < charArray.length; i++) {
                    char c = charArray[i];
                    Letter l = new Letter(i == charArray.length - 1);
                    if (!candidates.containsKey(c)) {
                        candidates.put(c, l);
                        candidates = l.children;
                    } else {
                        candidates = candidates.get(c).children;
                    }

                }

            }

        }
    }

    private static class Letter {
        private final boolean wordEnd;
        private Map<Character, Letter> children = new HashMap<>();

        public Letter(boolean wordEnd) {
            this.wordEnd = wordEnd;
        }
    }


X. DFS+cache
String bestSplit(HashSet<String> dictionary, String sentence) {
 ParseResult r = split(dictionary, sentence, 0);
 return r == null ? null : r.parsed;
}

ParseResult split(HashSet<String> dictionary, String sentence, int start) {
 if (start >= sentence.length())  return new ParseResult(0, "");

 int bestInvalid = Integer.MAX_VALUE;  // const value (2^31)-1
 String bestParsing = null;
 String partial = "";
 int index = start; // 0
 while (index < sentence.length()) {
  char c = sentence.charAt(index);
  partial += c;
  int invalid = dictionary.contains(partial) ? 0 : partial.length();
  if (invalid < bestInvalid) {
   ParseResult result = split(dictionary, sentence, index+1);
   if (invalid + result.invalid < bestInvalid) {
    bestInvalid = invalid + result.invalid;
    bestParsing = partial = " " + result.parsed;
    if (bestInvalid == 0) break;
   }
  }
  ++index;
 }
 return new ParseResult(bestIndex, bestParsing);

}

public class ParseResult {
 public int invalid = Integer.MAX_VALUE;
 public String parsed = " ";
 public ParseResult(int inv, String p) {
  invalid = inv;
  parsed = p;
 }
}


String split(HashSet<String> dictionary, String sentence, int start, ParseResult[] memo) {
 if (start >= sentence.length())  return new ParseResult(0, "");
 if (memo[start] != null)  return memo[start];
 int bestValid = Integer.MAX_VALUE;
 String bestParsing = null;
 String partial = "";
 int index = start;
 while (index < sentence.length()) {
  char c = sentence.chatAt(index);
  partial += c;
  int invalid = dictionary.contains(partial) ? 0 : partial.length();
  if (invalid < bestInvalid) {
   ParseResult result = split(dictionary, sentence, index+1, memo);
   if (invalid + result.invalid < bestInvalid) {
    bestInvalid = invalid + result.invalid;
    bestParsing = partial + " " + result.parsed;
    if (bestInvalid == 0)  break;
   }
  }
  ++index;
 }
 memo[start] = new ParseResult(bestInvalid, bestParsing);
 return memo[start];
}





我们需要拆分字符串,LeetCode中有类似的题目Word Break IIWord Break,但是又有不同,这道题允许有无法拆分的单词,那么对于这种问题,我们的目的是使无效的字母越少越好,基本的解法见parseSimple()函数,该函数可以有两点优化:
1. 一些递归重复计算了,我们可以使用哈希表来保存之前计算好的最优拆分了,那么在之后的递归中遇到了直接取出来用就行了。
2. 在有些情况下我们可以预测一种拆分方式是否会生成无效单词,比如当我们拆分单词xten,没有单词是以xt开头的,但是当前的算法会计算xt+p(en),xten+p(n)和xten。每一次我们都会发现这样的单词不存在,这样我们直接在x后面加一个空格就可以了,那么我们怎么样知道没有单词是以xt开始的呢,用前缀树trie,参见parseOptimized()函数。
我们使用哈希表来缓存结果,关键字是单词的起始位置,我们保存了最佳的位置来拆分余下的字符串。我们可以让代码返回拆分好的字符串,但是比较麻烦。我们使用一个外包类Result来返回无效字符的个数和最优字符串,而在C++中就不用这样,因为可以通过引用传递。
    public static String sentence;
    public static Trie dictionary;
    
    public static Result parse(int start, int end, Hashtable<Integer, Result> cache) {
        if (end >= sentence.length()) {
            return new Result(end - start, sentence.substring(start).toUpperCase());
        }
        if (cache.containsKey(start)) {
            return cache.get(start).clone();
        }
        String curWord = sentence.substring(start, end + 1);
        boolean validPartial = dictionary.contains(curWord, false);
        boolean validExact = validPartial && dictionary.contains(curWord, true);
        
        Result bestExact = parse(end + 1, end + 1, cache);
        if (validExact) {
            bestExact.parsed = curWord + " " + bestExact.parsed;
        } else {
            bestExact.invalid += curWord.length();
            bestExact.parsed = curWord.toUpperCase() + " " + bestExact.parsed;
        }
        
        Result bestExtend = null;
        if (validPartial) {
            bestExtend = parse(start, end + 1, cache);
        }
        
        Result best = Result.min(bestExact, bestExtend);
        cache.put(start, best.clone());
        return best;
    }
    
    public static int parseOptimized(int start, int end, Hashtable<Integer, Integer> cache) {
        if (end >= sentence.length()) {
            return end - start;
        }
        if (cache.containsKey(start)) {
            return cache.get(start);
        }
        String curWord = sentence.substring(start, end + 1);
        boolean validPartial = dictionary.contains(curWord, false);
        int bestExact = parseOptimized(end + 1, end + 1, cache);
        if (!validPartial || !dictionary.contains(curWord, true)) {
            bestExact += curWord.length();
        }
        int bestExtend = Integer.MAX_VALUE;
        if (validPartial) {
            bestExtend = parseOptimized(start, end + 1, cache);
        }
        int min = Math.min(bestExact, bestExtend);
        cache.put(start, min);
        return min;
    }
    
    public static int parseSimple(int start, int end) {
        if (end >= sentence.length()) {
            return end - start;
        }
        String word = sentence.substring(start, end + 1);
        int bestExact = parseSimple(end + 1, end + 1);
        if (!dictionary.contains(word, true)) {
            bestExact += word.length();
        }
        int bestExtend = parseSimple(start, end + 1);
        return Math.min(bestExact, bestExtend);
    }


The solution given in the book is very hard to understand. It uses HashMap to memorize the previous result.
After long time of struggle, I finally solved it with traditional DP approach. The key idea is to consider: "whether I insert a space after this char, or not".

dp[j] + i - j which means all words between i and j are not recognized.

public static int parse(String doc, Trie dict) {
    int len = doc.length();
    int[] dp = new int[len + 1];
    // dp[i] denotes the number of special chars in first i chars of docs
    for (int i = 1; i <= len; i++) {
        dp[i] =  Integer.MAX_VALUE;
        for (int j = 0; j < i; j++) {
            String str = doc.substring(j, i);
            if (dict.contains(str, true)) {
                // consider (i to j) a valid word
                dp[i] = Math.min(dp[i], dp[j]);
            } else {
                // consider (i to j) special chars
                dp[i] = Math.min(dp[i], dp[j] + i - j);
            }
        }
    }
    return dp[len];
}
http://www.cnblogs.com/EdwardLiu/p/4356967.html
 8     public int optway(String s, Set<String> dict) {
 9         if (s==null || s.length()==0) return 0;
10         int len = s.length();
11         if (dict.isEmpty()) return len;
12         int[] res = new int[len+1]; // res[i] refers to minimized # of unrecognized chars in first i chars
13         Arrays.fill(res, Integer.MAX_VALUE);
14         res[0] = 0;
15         for (int i=1; i<=len; i++) {
16             for (int j=0; j<i; j++) {
17                 String str = s.substring(j, i);
18                 int unrecogNum = dict.contains(str)? 0 : i-j;
19                 res[i] = Math.min(res[i], res[j]+unrecogNum);
20             }
21         }
22         return res[len];
23     }
Read full article from [CC150v5] 17.14 Optimal Way to Unconcatenate Doc - Shuatiblog.com

Labels

LeetCode (1432) GeeksforGeeks (1122) LeetCode - Review (1067) Review (882) Algorithm (668) to-do (609) Classic Algorithm (270) Google Interview (237) Classic Interview (222) Dynamic Programming (220) DP (186) Bit Algorithms (145) POJ (141) Math (137) Tree (132) LeetCode - Phone (129) EPI (122) Cracking Coding Interview (119) DFS (115) Difficult Algorithm (115) Lintcode (115) Different Solutions (110) Smart Algorithm (104) Binary Search (96) BFS (91) HackerRank (90) Binary Tree (86) Hard (79) Two Pointers (78) Stack (76) Company-Facebook (75) BST (72) Graph Algorithm (72) Time Complexity (69) Greedy Algorithm (68) Interval (63) Company - Google (62) Geometry Algorithm (61) Interview Corner (61) LeetCode - Extended (61) Union-Find (60) Trie (58) Advanced Data Structure (56) List (56) Priority Queue (53) Codility (52) ComProGuide (50) LeetCode Hard (50) Matrix (50) Bisection (48) Segment Tree (48) Sliding Window (48) USACO (46) Space Optimization (45) Company-Airbnb (41) Greedy (41) Mathematical Algorithm (41) Tree - Post-Order (41) ACM-ICPC (40) Algorithm Interview (40) Data Structure Design (40) Graph (40) Backtracking (39) Data Structure (39) Jobdu (39) Random (39) Codeforces (38) Knapsack (38) LeetCode - DP (38) Recursive Algorithm (38) String Algorithm (38) TopCoder (38) Sort (37) Introduction to Algorithms (36) Pre-Sort (36) Beauty of Programming (35) Must Known (34) Binary Search Tree (33) Follow Up (33) prismoskills (33) Palindrome (32) Permutation (31) Array (30) Google Code Jam (30) HDU (30) Array O(N) (29) Logic Thinking (29) Monotonic Stack (29) Puzzles (29) Code - Detail (27) Company-Zenefits (27) Microsoft 100 - July (27) Queue (27) Binary Indexed Trees (26) TreeMap (26) to-do-must (26) 1point3acres (25) GeeksQuiz (25) Merge Sort (25) Reverse Thinking (25) hihocoder (25) Company - LinkedIn (24) Hash (24) High Frequency (24) Summary (24) Divide and Conquer (23) Proof (23) Game Theory (22) Topological Sort (22) Lintcode - Review (21) Tree - Modification (21) Algorithm Game (20) CareerCup (20) Company - Twitter (20) DFS + Review (20) DP - Relation (20) Brain Teaser (19) DP - Tree (19) Left and Right Array (19) O(N) (19) Sweep Line (19) UVA (19) DP - Bit Masking (18) LeetCode - Thinking (18) KMP (17) LeetCode - TODO (17) Probabilities (17) Simulation (17) String Search (17) Codercareer (16) Company-Uber (16) Iterator (16) Number (16) O(1) Space (16) Shortest Path (16) itint5 (16) DFS+Cache (15) Dijkstra (15) Euclidean GCD (15) Heap (15) LeetCode - Hard (15) Majority (15) Number Theory (15) Rolling Hash (15) Tree Traversal (15) Brute Force (14) Bucket Sort (14) DP - Knapsack (14) DP - Probability (14) Difficult (14) Fast Power Algorithm (14) Pattern (14) Prefix Sum (14) TreeSet (14) Algorithm Videos (13) Amazon Interview (13) Basic Algorithm (13) Codechef (13) Combination (13) Computational Geometry (13) DP - Digit (13) LCA (13) LeetCode - DFS (13) Linked List (13) Long Increasing Sequence(LIS) (13) Math-Divisible (13) Reservoir Sampling (13) mitbbs (13) Algorithm - How To (12) Company - Microsoft (12) DP - Interval (12) DP - Multiple Relation (12) DP - Relation Optimization (12) LeetCode - Classic (12) Level Order Traversal (12) Prime (12) Pruning (12) Reconstruct Tree (12) Thinking (12) X Sum (12) AOJ (11) Bit Mask (11) Company-Snapchat (11) DP - Space Optimization (11) Dequeue (11) Graph DFS (11) MinMax (11) Miscs (11) Princeton (11) Quick Sort (11) Stack - Tree (11) 尺取法 (11) 挑战程序设计竞赛 (11) Coin Change (10) DFS+Backtracking (10) Facebook Hacker Cup (10) Fast Slow Pointers (10) HackerRank Easy (10) Interval Tree (10) Limited Range (10) Matrix - Traverse (10) Monotone Queue (10) SPOJ (10) Starting Point (10) States (10) Stock (10) Theory (10) Tutorialhorizon (10) Kadane - Extended (9) Mathblog (9) Max-Min Flow (9) Maze (9) Median (9) O(32N) (9) Quick Select (9) Stack Overflow (9) System Design (9) Tree - Conversion (9) Use XOR (9) Book Notes (8) Company-Amazon (8) DFS+BFS (8) DP - States (8) Expression (8) Longest Common Subsequence(LCS) (8) One Pass (8) Quadtrees (8) Traversal Once (8) Trie - Suffix (8) 穷竭搜索 (8) Algorithm Problem List (7) All Sub (7) Catalan Number (7) Cycle (7) DP - Cases (7) Facebook Interview (7) Fibonacci Numbers (7) Flood fill (7) Game Nim (7) Graph BFS (7) HackerRank Difficult (7) Hackerearth (7) Inversion (7) Kadane’s Algorithm (7) Manacher (7) Morris Traversal (7) Multiple Data Structures (7) Normalized Key (7) O(XN) (7) Radix Sort (7) Recursion (7) Sampling (7) Suffix Array (7) Tech-Queries (7) Tree - Serialization (7) Tree DP (7) Trie - Bit (7) 蓝桥杯 (7) Algorithm - Brain Teaser (6) BFS - Priority Queue (6) BFS - Unusual (6) Classic Data Structure Impl (6) DP - 2D (6) DP - Monotone Queue (6) DP - Unusual (6) DP-Space Optimization (6) Dutch Flag (6) How To (6) Interviewstreet (6) Knapsack - MultiplePack (6) Local MinMax (6) MST (6) Minimum Spanning Tree (6) Number - Reach (6) Parentheses (6) Pre-Sum (6) Probability (6) Programming Pearls (6) Rabin-Karp (6) Reverse (6) Scan from right (6) Schedule (6) Stream (6) Subset Sum (6) TSP (6) Xpost (6) n00tc0d3r (6) reddit (6) AI (5) Abbreviation (5) Anagram (5) Art Of Programming-July (5) Assumption (5) Bellman Ford (5) Big Data (5) Code - Solid (5) Code Kata (5) Codility-lessons (5) Coding (5) Company - WMware (5) Convex Hull (5) Crazyforcode (5) DFS - Multiple (5) DFS+DP (5) DP - Multi-Dimension (5) DP-Multiple Relation (5) Eulerian Cycle (5) Graph - Unusual (5) Graph Cycle (5) Hash Strategy (5) Immutability (5) Java (5) LogN (5) Manhattan Distance (5) Matrix Chain Multiplication (5) N Queens (5) Pre-Sort: Index (5) Quick Partition (5) Quora (5) Randomized Algorithms (5) Resources (5) Robot (5) SPFA(Shortest Path Faster Algorithm) (5) Shuffle (5) Sieve of Eratosthenes (5) Strongly Connected Components (5) Subarray Sum (5) Sudoku (5) Suffix Tree (5) Swap (5) Threaded (5) Tree - Creation (5) Warshall Floyd (5) Word Search (5) jiuzhang (5)

Popular Posts