题目:
给你一个字符串 s
和一个字符串列表 wordDict
作为字典。如果可以利用字典中出现的一个或多个单词拼接出 s
则返回 true
。
注意:不要求字典中出现的单词全部都使用,并且字典中的单词可以重复使用。
示例 1:
输入: s = "leetcode", wordDict = ["leet", "code"] 输出: true 解释: 返回 true 因为 "leetcode" 可以由 "leet" 和 "code" 拼接成。
示例 2:
输入: s = "applepenapple", wordDict = ["apple", "pen"] 输出: true 解释: 返回 true 因为 "applepenapple" 可以由 "apple" "pen" "apple" 拼接成。 注意,你可以重复使用字典中的单词。
示例 3:
输入: s = "catsandog", wordDict = ["cats", "dog", "sand", "and", "cat"] 输出: false
提示:
1 <= s.length <= 300
1 <= wordDict.length <= 1000
1 <= wordDict[i].length <= 20
s
和wordDict[i]
仅由小写英文字母组成wordDict
中的所有字符串互不相同
1.确定dp数组的含义
dp[i] : 字符串长度为i的话,dp[i]为true,表示可以拆分为一个或多个在字典中出现的单词。
2.确定递推公式
如果确定dp[j] 是true,且 [j, i] 这个区间的子串出现在字典里,那么dp[i]一定是true。(j < i )。
所以递推公式是 if([j, i] 这个区间的子串出现在字典里 && dp[j]是true) 那么 dp[i] = true。
3.dp数组如何初始化
从递推公式中可以看出,dp[i] 的状态依靠 dp[j]是否为true,那么dp[0]就是递推的根基,dp[0]一定要为true,否则递推下去后面都都是false了。
4.确定遍历顺序
题目中说是拆分为一个或多个在字典中出现的单词,所以这是完全背包。此题求的使排列数,求排列数就是外层for遍历背包,内层for循环遍历物品。
实例代码:
class Solution {
public:
bool wordBreak(string s, vector<string>& wordDict) {
//典型的完全背包问题 字符串要求有顺序 排列数 先遍历背包在遍历物品
vector<bool> dp(s.size()+1,false);
dp[0]=true;//dp[0]表示空字符串 以便于推导递推公式
for(int i=1;i<=s.size();i++)
{
for(int j=0;j<i;j++)
{
string a=s.substr(j,i-j);//从j开始 截取i-j长度的字符串
auto it = find(wordDict.begin(), wordDict.end(), a);
if(it!=wordDict.end()&&dp[j]==true){//查找该段元素是否出现在物品里
dp[i]=true;
break;
}
}
}
return dp[s.size()];
}
};