給定一個字符串s和單詞字典dict,肯定s是否能夠分割成一個或多個單詞空格分隔的序列。java
一個字符串S,它的長度爲N,若是S可以被「字典集合」(dict)中的單詞拼接而成,那麼所要知足的條件爲:
* F(0, N) = F(0, i) && F(i, j) && F(j, N);
* 這樣子,若是咱們想知道某個子串是否可由Dict中的幾個單詞拼接而成就能夠用這樣的方式獲得結果(知足條件爲True, 不知足條件爲False)存入到一個boolean數組的對應位置上,這樣子,最後boolean 數組的最後一位就是F(0, N)的值,爲True表示這個字符串S可由Dict中的單詞拼接,不然不行!算法
算法實現類數組
import java.util.HashMap; import java.util.HashSet; import java.util.Map; import java.util.Set; public class Solution { public boolean wordBreak(String s, Set<String> wordDict) { // 參數校驗 if (s == null || s.length() < 1 || wordDict == null || wordDict.size() < 1) { return false; } // 標記是否匹配,match[i]表表[0, i-1]都匹配 int length = s.length(); boolean[] match = new boolean[length + 1]; match[0] = true; for (int i = 1; i < length + 1; i++) { for (int j = 0; j < i; j++) { if (match[j] && wordDict.contains(s.substring(j, i))) { // f(0,n) = f(0,i) + f(i,j) + f(j,n) match[i] = true; break; } } } return match[length]; } // 下面是另外一種解法,可是會超時 public boolean wordBreak2(String s, Set<String> wordDict) { // 參數校驗 if (s == null || s.length() < 1 || wordDict == null || wordDict.size() < 1) { return false; } Map<Character, Set<String>> wordMap = new HashMap<>(wordDict.size()); // 將全部開始字符相同的單詞放入一個Set中 for (String word : wordDict) { Set<String> set = wordMap.get(word.charAt(0)); if (set == null) { // 新建立一個set放入Map中 set = new HashSet<>(); wordMap.put(word.charAt(0), set); } // 單詞存入set中 set.add(word); } return wordBreak(s, 0, wordMap); } /** * 搜索字符串是否能夠被分割成單詞串 * * @param s 字符串 * @param idx 處理的開始位置 * @param wordMap 單詞字典,開始字符相同的在同一個set集合中 * @return 搜索結果 */ public boolean wordBreak(String s, int idx, Map<Character, Set<String>> wordMap) { if (idx >= s.length()) { return true; } Set<String> words = wordMap.get(s.charAt(idx)); if (words != null) { for (String word : words) { // idx以前的字符已經匹配,若是從ide以後起匹配word單詞 if (s.startsWith(word, idx)) { // 遞歸處理 boolean result = wordBreak(s, idx + word.length(), wordMap); // 若是知足條件,返回true if (result) { return true; } } } } return false; } }