LeetCode刷题笔记 - JavaScript(六)

剑指 Offer II 005. 单词长度的最大乘积
剑指 Offer II 006. 排序数组中两个数字之和
剑指 Offer II 007. 数组中和为 0 的三个数

1.剑指 Offer II 005. 单词长度的最大乘积

给定一个字符串数组 words,请计算当两个字符串 words[i] 和 words[j] 不包含相同字符时,它们长度的乘积的最大值。假设字符串中只包含英语的小写字母。如果没有不包含相同字符的一对字符串,返回 0。

题目大意:求字符串数组中两个完全不相同的字符串长度乘积的最大值。

解题思路:位运算。定义hashs数组存储每一个单词的字符hash值,对两个单词的hash值进行按位与&运算,即可判断两个单词的字母是否完全相同,然后在此期间,记录最大乘积即可。

代码

var maxProduct = function(words) {
    const lengths = [];
    const hashs = new Array(words.length).fill(0);
    let res = 0;
    for(let i = 0; i < words.length; i++) {
        lengths[i] = words[i].length;
        for(let ch of words[i]) {
            hashs[i] |= 1 << ch.charCodeAt() - 'a'.charCodeAt();
        }
        for(let j = 0; j < i; j++) {
            if((hashs[j] & hashs[i]) === 0) {
                res = Math.max(res, lengths[i] * lengths[j]);
            }
        }
    }
    return res;
};

2.剑指 Offer II 006. 排序数组中两个数字之和

给定一个已按照 升序排列 的整数数组 numbers ,请你从数组中找出两个数满足相加之和等于目标数 target 。
函数应该以长度为 2 的整数数组的形式返回这两个数的下标值。numbers 的下标 从 0 开始计数 ,所以答案数组应当满足 0 <= answer[0] < answer[1] < numbers.length 。

题目大意:求一个升序数组中和为target的两个数的下标。

解题思路:双指针写法。因为数组是升序排序,因此只需要使用两个指针l和r,分别指向数组的首部和尾部,若当前首部和尾部的和>target,使尾指针向左移动,否则,若和<target,则头指针向右移动。

代码

var twoSum = function(numbers, target) {
    let i = 0, j = numbers.length - 1;
    while(i < j) {
        const res = numbers[i] + numbers[j];
        if(res > target) j--;
        else if(res < target) i++;
        else return [i,j];
    }
};

3.剑指 Offer II 007. 数组中和为 0 的三个数

给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i != j、i != k 且 j != k ,同时还满足 nums[i] + nums[j] + nums[k] == 0 。请你返回所有和为 0 且不重复的三元组。

题目大意:求一个数组中,所有和为0的三个数

解题思路:此题若使用暴力解法,将会收获O(n^3)的时间复杂度和用于去重的map所占用的大量空间复杂度。先考虑去重的问题,首先,对于原数组排序,在三层循环中,需要判断当前循环变量与上一个循环变量的值不相等,否则跳过当前循环。这样就解决了重复值的问题。再来优化时间复杂度,第一层循环我们仍然遍历nums数组,然后内存循环因为我们已经对数组进行了排序,因此可以采用双指针的写法,将时间复杂度优化为O(n^2)。
代码

var threeSum = function(nums) {
    nums.sort((a,b) => a-b);
    const res = [];
    for(let i = 0; i < nums.length-2; i++) {
        if(i && nums[i] === nums[i-1])continue;
        let l = i + 1, r = nums.length - 1;
        const target = -nums[i];
        while(l < r) {
            const sum = nums[l] + nums[r];
            if(sum > target) {
                r--;
            } else if(sum < target) {
                l++;
            } else {
                if(!(l > i+1 && nums[l] === nums[l-1])) res.push([nums[i],nums[l],nums[r]]);
                l++;
                r--;
            }
        }
    }
    return res;
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

volit_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值