-
leetcode-472-连接词
题目要理解起来不难,重点在于怎么判断一个字符串是否可以由其他的几个字符串组成。
既然是由其他的字符串组成的,那么肯定是由比它更短的几个组成的,这里用一个记录所有的单词,主要是为了防止由空串这种特例,因为不能由自己来构成自己。
剩下来的步骤就是判断是否可以由里的单词来组成自身,如果可以则加入结果中。
判断的过程如下:
- 如果集合里没有单词或者是这个单词本身就是空串,那么肯定无法组成这个单词;
- 构造数组,的含义为之间的子串是否可以分解为在中的单词,就代表是否可以完全分解;
- 那么就可以根据如上判断是否可以分解为中的较短单词,先将上式分为和,前者对应了,后者则为,只需判断是否包含有即可,如果包含则;
- 返回
class Solution {
public List<String> findAllConcatenatedWordsInADict(String[] words) {
List<String> ans = new ArrayList<>();
int n = words.length;
if(n < 3) {
return ans;
}
Set<String> set = new HashSet<>(Arrays.asList(words));
for(int i = 0; i < n; i++) {
// 去重空串
if("".equals(words[i])) {
continue;
}
// 先去重本身,后面再加回来
set.remove(words[i]);
if(canBreak(words[i], set)) {
ans.add(words[i]);
}
set.add(words[i]);
}
return ans;
}
// 判断word是否可由set中的单词组合而成
private boolean canBreak(String word, Set<String> set) {
int len = word.length();
// 如果单词长度为0或者是集合为空了,那么肯定无法组成这个单词
if(len == 0 || set.size() == 0) {
return false;
}
boolean[] dp = new boolean[len + 1];
dp[0] = true;
for(int i = 1; i < len + 1; i++) {
for(int j = 0; j < i; j++) {
// 如果无法构成word[0, j- 1]之间跳过
if(!dp[j]) {
continue;
}
if(set.contains(word.substring(j, i))) {
dp[i] = true;
break;
}
}
}
return dp[len];
}
}
除了上述的做法,还可以利用字典树来减少前缀的匹配过程,这样对于每个单词,就不要遍历完所有的字符。
我们可以先把单词数组按照长度从小到大排序,这样就可以边检测边判断是否有连接词,因为连接词是由其他单词组成的,所以它的长度肯定更长,在后面才做匹配过程,如果不是连接词再把它加入到字典树中去。
class Solution {
// 定义trie
private static class TrieNode {
boolean isWord;
Map<Character, TrieNode> children;
TrieNode() {
this.isWord = false;
this.children = new HashMap<>();
}
}
// 将单词插入到字典树中
private void insert(String word) {
char[] letters = word.toCharArray();
TrieNode curr = root;
for(char letter : letters) {
if(curr.children.get(letter) == null) {
curr.children.put(letter, new TrieNode());
}
curr = curr.children.get(letter);
}
curr.isWord = true;
}
// 字典树的根节点
TrieNode root = new TrieNode();
public List<String> findAllConcatenatedWordsInADict(String[] words) {
List<String> ans = new ArrayList<>();
Arrays.sort(words, (a, b) -> a.length() - b.length());
for(String word : words) {
if(!word.isBlank()) {
// 可以由字典树中的单词组成则返回结果
if(dfs(word, 0)) {
ans.add(word);
} else {
// 不是连接词的则插入字典树中
insert(word);
}
}
}
return ans;
}
private boolean dfs(String word, int position) {
if(position == word.length()) {
return true;
}
TrieNode curr = this.root;
while(position < word.length()) {
// 字典树中不存在该字符
if(curr.children.get(word.charAt(position)) == null) {
return false;
}
curr = curr.children.get(word.charAt(position));
// 如果形成了一个完整的单词,则进入下一层寻找是否可继续组成
if(curr.isWord && dfs(word, position + 1)) {
return true;
}
position++;
}
return false;
}
}
__EOF__

最新更新
求1000阶乘的结果末尾有多少个0
详解MyBatis延迟加载是如何实现的
IDEA 控制台中文乱码4种解决方案
SpringBoot中版本兼容性处理的实现示例
Spring的IOC解决程序耦合的实现
详解Spring多数据源如何切换
Java报错:UnsupportedOperationException in Col
使用Spring Batch实现批处理任务的详细教程
java中怎么将多个音频文件拼接合成一个
SpringBoot整合ES多个精确值查询 terms功能实
数据库审计与智能监控:从日志分析到异
SQL Server 中的数据类型隐式转换问题
SQL Server中T-SQL 数据类型转换详解
sqlserver 数据类型转换小实验
SQL Server数据类型转换方法
SQL Server 2017无法连接到服务器的问题解决
SQLServer地址搜索性能优化
Sql Server查询性能优化之不可小觑的书签查
SQL Server数据库的高性能优化经验总结
SQL SERVER性能优化综述(很好的总结,不要错
uniapp/H5 获取手机桌面壁纸 (静态壁纸)
[前端] DNS解析与优化
为什么在js中需要添加addEventListener()?
JS模块化系统
js通过Object.defineProperty() 定义和控制对象
这是目前我见过最好的跨域解决方案!
减少回流与重绘
减少回流与重绘
如何使用KrpanoToolJS在浏览器切图
performance.now() 与 Date.now() 对比