【每日一题Day370】LC318最大单词长度乘积 | 哈希表 位运算

最大单词长度乘积【LC318】

给你一个字符串数组 words ,找出并返回 length(words[i]) * length(words[j]) 的最大值,并且这两个单词不含有公共字母。如果不存在这样的两个单词,返回 0

2022/10/17

位运算

  1. 将每个单词转化为整数(二进制形式),用最低位代表a,第二低位代表b......最高位(第二十六位)代表z,例如单词"ab"可表示为11,十进制为3
  2. 将某个单词的二进制形式与其他单词进行与运算,如果结果为0,那么代表这两个单词不含有公共字母,判断是否需要更新结果
  • 代码

    java 复制代码
    class Solution {
        public int maxProduct(String[] words) {
            int len = words.length;
            int[] binary = new int[len]; // a:1 b:2 c:4 ......
            for (int i = 0; i < len; i++){
                String str = words[i];
                boolean[] map = new boolean[26];
                for (int j = 0; j < str.length(); j++){
                    int n = str.charAt(j)-'a';
                    if (!map[n]){
                        binary[i] += Math.pow(2,n);
                        map[n] = true;
                    }
                }
            }
            int res = 0;
            for (int i = 0; i < len; i++){
                for (int j = i + 1; j < len; j++){
                    if ((binary[i] & binary[j]) == 0){
                        res = Math.max(res,words[i].length()*words[j].length());
                    }
                }
            }    
            return res;    
        }
    }
  • 复杂度

    • 时间复杂度:O(n2+nk),n为字符串的个数,k为每个字符串的平均长度

    • 空间复杂度:O(n)

  • 优化

    将每个字符串转化为二进制的代码可优化为

    java 复制代码
    for (int i = 0; i < len; i++){
         String str = words[i];
         for (char ch: words[i].toCharArray()){
            binary[i] |= 1 << (ch - 'a');	
        }
    }

哈希表

使用哈希表记录出现在该字符串的所有字符

  • 代码

    java 复制代码
    class Solution {
        public int maxProduct(String[] words) {
            int len = words.length;
            boolean[][] flags = new boolean[len][26]; 
            for (int i = 0; i < len; i++){
                String str = words[i];
                for (char ch: words[i].toCharArray()){
                    flags[i][ch-'a'] = true;
                }
            }
            int res = 0;
            for (int i = 0; i < len; i++){
                for (int j = i + 1; j < len; j++){
                    int k = 0;
                    for (; k < 26; k++){
                        if (flags[i][k] && flags[j][k]){
                            break;
                        }
                    }
                    if (k == 26){
                        res = Math.max(res,words[i].length()*words[j].length());
                    }
                }
            }    
            return res;    
        }
    }
  • 复杂度

    • 时间复杂度:O(n2+nk),n为字符串的个数,k为每个字符串的平均长度

    • 空间复杂度:O(n)

相关推荐
老歌老听老掉牙8 分钟前
使用 SymPy 进行向量和矩阵的高级操作
python·线性代数·算法·矩阵·sympy
lifallen17 分钟前
Flink checkpoint
java·大数据·算法·flink
比特森林探险记31 分钟前
Go 中的 Map 与字符处理指南
c++·算法·golang
安全系统学习2 小时前
网络安全逆向分析之rust逆向技巧
前端·算法·安全·web安全·网络安全·中间件
sz66cm2 小时前
LeetCode刷题 -- 542. 01矩阵 基于 DFS 更新优化的多源最短路径实现
leetcode·矩阵·深度优先
菜鸟懒懒3 小时前
exp1_code
算法
Winn~3 小时前
JVM垃圾回收器-ZGC
java·jvm·算法
爱coding的橙子4 小时前
每日算法刷题Day24 6.6:leetcode二分答案2道题,用时1h(下次计时20min没写出来直接看题解,节省时间)
java·算法·leetcode
慢慢慢时光4 小时前
leetcode sql50题
算法·leetcode·职场和发展
pay顿4 小时前
力扣LeetBook数组和字符串--二维数组
算法·leetcode