给你一个字符串数组 words ,找出并返回 length(words[i]) * length(words[j]) 的最大值,并且这两个单词不含有公共字母。如果不存在这样的两个单词,返回 0。
示例 1:
输入:words = [“abcw”,“baz”,“foo”,“bar”,“xtfn”,“abcdef”]
输出:16
解释:这两个单词为 “abcw”, “xtfn”。
示例 2:
输入:words = [“a”,“ab”,“abc”,“d”,“cd”,“bcd”,“abcd”]
输出:4
解释:这两个单词为 “ab”, “cd”。
示例 3:
输入:words = [“a”,“aa”,“aaa”,“aaaa”]
输出:0
解释:不存在这样的两个单词。
提示:
2 <= words.length <= 1000
1 <= words[i].length <= 1000
words[i] 仅包含小写字母
(1)暴力穷举法 1
注意:该方法在 LeetCode 中提交时会出现“超出时间限制”的提示!
(2)暴力穷举法 2
(3)位运算
相较于之前的两段代码,这段代码在存储单词的字符信息时使用了位运算,将字符掩码存储在整型数组中。这样做的好处是可以通过按位与运算快速判断两个单词是否有公共字符。
//思路1————暴力穷举法 1
class Solution {
public int maxProduct(String[] words) {
//按照单词长度对 words 进行降序排序
Arrays.sort(words, (w1, w2) -> (w2.length() - w1.length()));
int res = 0;
for (int i = 0; i < words.length; i++) {
for (int j = 0; j < i + 1; j++) {
if (check(words[i], words[j])) {
int tmp = words[i].length() * words[j].length();
if (res < tmp) {
res = tmp;
break;
}
}
}
}
return res;
}
//判断单词 w1 和 w2 是否有公共字母
public boolean check(String w1, String w2) {
Set<Character> set = new HashSet<>();
for (int i = 0; i < w1.length(); i++) {
set.add(w1.charAt(i));
}
for (int i = 0; i < w2.length(); i++) {
if (set.contains(w2.charAt(i))) {
return false;
}
}
return true;
}
}
//思路2————暴力穷举法 2
class Solution {
public int maxProduct(String[] words) {
int n = words.length;
HashSet set[] = new HashSet[n];
for (int i = 0; i < n; i++) {
set[i] = new HashSet<Character>();
}
for (int i = 0; i < words.length; i++) {
for (int j = 0; j < words[i].length(); j++) {
set[i].add(words[i].charAt(j));
}
}
int res = 0;
for (int t = 0; t < set.length; t++) {
for (int j = t + 1; j < set.length; j++) {
int flag = 0;
for (Character c : (HashSet<Character>) set[t]) {
if (set[j].contains(c)) {
flag = 1;
break;
}
}
if (flag == 0) {
res = Math.max(res, words[t].length() * words[j].length());
}
}
}
return res;
}
}
//思路3————位运算
class Solution {
public int maxProduct(String[] words) {
int length = words.length;
int[] masks = new int[length];
for (int i = 0; i < length; i++) {
String word = words[i];
int wordLength = word.length();
for (int j = 0; j < wordLength; j++) {
masks[i] |= 1 << (word.charAt(j) - 'a');
}
}
int res = 0;
for (int i = 0; i < length; i++) {
for (int j = i + 1; j < length; j++) {
if ((masks[i] & masks[j]) == 0) {
res = Math.max(res, words[i].length() * words[j].length());
}
}
}
return res;
}
}