【每日一题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)

相关推荐
Imxyk17 小时前
力扣:632. 最小区间(贪心)
java·数据结构·算法
Mr_Xuhhh17 小时前
递归和迭代的区别(C/C++实现)
算法
历程里程碑17 小时前
21:重谈重定义理解一切皆“文件“及缓存区
linux·c语言·开发语言·数据结构·c++·算法·缓存
2501_9011478317 小时前
PyTorch DDP官方文档学习笔记(核心干货版)
pytorch·笔记·学习·算法·面试
Daydream.V17 小时前
决策树三中分类标准
算法·决策树·分类
闲人不梦卿17 小时前
数据结构之排序方法
数据结构·算法·排序算法
TracyCoder12317 小时前
LeetCode Hot100(24/100)——21. 合并两个有序链表
算法·leetcode·链表
power 雀儿17 小时前
前馈网络+层归一化
人工智能·算法
爱吃rabbit的mq17 小时前
第10章:支持向量机:找到最佳边界
算法·机器学习·支持向量机
木非哲17 小时前
AB实验高级必修课(四):逻辑回归的“马甲”、AUC的概率本质与阈值博弈
算法·机器学习·逻辑回归·abtest