当前位置: 首页 > news >正文

代码随想录算法训练营第7天

454.四数相加

题目链接:454. 四数相加 II - 力扣(LeetCode)

视频/文档链接:代码随想录 (programmercarl.com)

第一想法

遍历数组num1,num2,计算其和出现的数量,放入map集合中,键为和,值为出现的次数。遍历num3,num4,0-若其和的值出现在Map集合中,则count+=该值即可。

可优化点

不熟悉调用mapAPI。

map.put(sum, map.getOrDefault(sum, 0) + 1);
res += map.getOrDefault(0 - i - j, 0);

代码

class Solution {public int fourSumCount(int[] nums1, int[] nums2, int[] nums3, int[] nums4) {int count = 0;Map<Integer, Integer> map = new HashMap<Integer, Integer>();for(int i : nums1){for(int j:nums2){int sum = i+j;map.put(sum,map.getOrDefault(sum,0)+1);}}for (int k : nums3) {for (int n : nums4) {int sum = k+n;if(map.containsKey(-sum)){count+=map.get(-sum);}}}return count;}
}

383.赎金信

 题目链接:383. 赎金信 - 力扣(LeetCode)

视频/文档链接:代码随想录 (programmercarl.com)

第一想法

这个题和力扣242题很像。

对于案例:ransomNote = "aa", magazine = "aab"

力扣242:字符串s和t每个字母出现的次数都必须一样。所以返回false

力扣383:只要求ransdomNote每个字母数量<=magazine,则返回true。

代码

一次过。

class Solution {public boolean canConstruct(String ransomNote, String magazine) {int[] ransdomArr = new int[26];int[] magazineArr = new int[26];for (int i = 0; i < ransomNote.length(); i++) {ransdomArr[ransomNote.charAt(i)-'a']++;}for (int j = 0; j < magazine.length(); j++) {magazineArr[magazine.charAt(j)-'a']++;}for (int i = 0; i < 26; i++) {if(ransdomArr[i]>magazineArr[i])return false;}return true;}
}

15.三数之和

题目链接:15. 三数之和 - 力扣(LeetCode)

文档/视频链接:代码随想录 (programmercarl.com)

第一想法

a+b+c = 0,直接暴力循环然后去重。
a从第1个元素遍历,b从i+1处遍历,然后c的值应为-(a+b),看map集合中是否存在。最终去除重复元素。

代码随想录想法

首先先将数组排序,一层for循环,i代表当前元素a,再定义left指针b和right指针c,

若相加之和>0,则right--,相加之和<0,left++,相加之和 = 0,则加入结果集。直到left = right终止循环,相等时无意义。

当去重a时需要判断 if(nums[i] != nums[i-1]),即当前元素与之前元素不同,那么第一个元素不是会报空指针异常么,怎么避免这个问题?按照这样就能避免对 i= 0 时进行判定了。

  if (i > 0 && nums[i] == nums[i - 1]) {  // 去重acontinue;}

每次遍历对第一个元素要都判断是否>0,所以nums[i]>0这个判断逻辑要加到循环里面。还有个细节是直接return,而不是continue了。因为已经排序的数组,一旦碰上正数,往后的也必须是正数。

关于b和c的去重

我在看视频的时候也想的是直接将去重逻辑提前。

 if (nums[i] + nums[left] + nums[right] > 0) {
        right--;
        // 去重 right
        while (left < right && nums[right] == nums[right + 1]) right--;
    } else if (nums[i] + nums[left] + nums[right] < 0) {
        left++;
        // 去重 left
        while (left < right && nums[left] == nums[left - 1]) left++;
    } else {
    }

但对提升效率没有帮助,反正去重也是一个一个减下去,在哪里减都是一样的。

索性放在else中,结构更好看一点。

同时去重时还要判断left<right,是为了防止指针越界需要加上。可以记小模版就是while(left<right)再套while循环,里面while条件要考虑加外层循环条件。

代码

class Solution {public List<List<Integer>> threeSum(int[] nums) {List<List<Integer>> endResult = new ArrayList<>();Arrays.sort(nums);//排序for (int i = 0; i < nums.length; i++) {if(nums[i]>0)return endResult;if(i>0&&nums[i]==nums[i-1])continue;//进入判断逻辑int left = i + 1;int right = nums.length - 1;while (left < right){if(nums[i]+nums[left]+nums[right]>0)right--;else if(nums[i]+nums[left]+nums[right]<0)left++;else{//将结果加入endResult中,这个语句是这样写的。endResult.add(Arrays.asList(nums[i],nums[left],nums[right]));//这里就需要去重了while (left<right&&nums[right-1]==nums[right])right--;//假设2,3,3,循环结束后right指的是最后一个重复元素3(第2个)while (left<right&&nums[left+1]==nums[left])left++;//同理right--;//这里才正真跳到最终不同的值上left++;}}}return endResult;}
}

18.四数之和

题目链接/文章讲解/视频讲解: 代码随想录

在三数之和基础上再套一层循环。

看完代码随想录的想法

剪枝的逻辑不能是nums[i]>0,因为此时目标是target,而target可能是负数,排序之后的数组可以是越加越小,即target = -5, nums = [-4,-1,0,0]。

但是我们依旧可以去做剪枝,逻辑变成nums[i] > target && (nums[i] >=0 || target >= 0)就可以了。

 每次对第一个元素不做判断,所以第二层循环去重时j>i+1&&nums[j]==nums[j-1],而不应当惯性思维是j>0

class Solution {public List<List<Integer>> fourSum(int[] nums, int target) {List<List<Integer>> endResult = new ArrayList<>();Arrays.sort(nums);for (int i = 0; i < nums.length; i++) {//剪枝if(nums[i]>target&&(nums[i]>0||target>0))return endResult;if(i>0&&nums[i]==nums[i-1])continue;for (int j = i + 1; j < nums.length - 1 - 1; j++) { //0,1,2,3,j<2,//去重if( j > i+1 &&nums[j] == nums [j-1]) //j为1时不应去重continue;int left = j+1;int right = nums.length-1;while (left<right){if(nums[i]+nums[j]+nums[left]+nums[right]>target)right--;else if(nums[i]+nums[j]+nums[left]+nums[right]<target)left++;else{endResult.add(Arrays.asList(nums[i],nums[j],nums[left],nums[right]));while (left<right&&nums[right-1] ==nums[right])right--;while (left<right&&nums[left+1] == nums[left])left++;right--;left++;}}}}return endResult;}
}

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • Mybatis——增删改查
  • Django学习收尾
  • 7.9实验室总结 SceneBuilder的使用方法+使用javafx等
  • 【Linux】:程序替换
  • 这不是在搞技术,而是在玩心态~
  • JS进阶-深入对象
  • 音视频封装demo:将h264数据和aac数据封装(mux)成FLV文件(纯手工,不依赖第三方开源库)
  • 面试题007-Java-Spring
  • 华为机试真题--字符串变换最小字符串
  • 初识STM32:寄存器编程 × 库函数编程 × 开发环境
  • ubuntu下aarch64-linux-gnu(交叉编译) gdb/gdbserver
  • 如何从数码相机恢复已删除的照片
  • Python开发—— 列表的高级操作与应用
  • spring监听事件
  • Obsidian 文档编辑器
  • 「前端」从UglifyJSPlugin强制开启css压缩探究webpack插件运行机制
  • 【翻译】Mashape是如何管理15000个API和微服务的(三)
  • Android Volley源码解析
  • Java到底能干嘛?
  • JSDuck 与 AngularJS 融合技巧
  • Laravel深入学习6 - 应用体系结构:解耦事件处理器
  • open-falcon 开发笔记(一):从零开始搭建虚拟服务器和监测环境
  • 力扣(LeetCode)21
  • 排序算法之--选择排序
  • 区块链分支循环
  • 网页视频流m3u8/ts视频下载
  • 智能网联汽车信息安全
  • 【云吞铺子】性能抖动剖析(二)
  • 国内唯一,阿里云入选全球区块链云服务报告,领先AWS、Google ...
  • ​​​​​​​​​​​​​​Γ函数
  • ​flutter 代码混淆
  • ​LeetCode解法汇总2583. 二叉树中的第 K 大层和
  • ​埃文科技受邀出席2024 “数据要素×”生态大会​
  • ​七周四次课(5月9日)iptables filter表案例、iptables nat表应用
  • ​软考-高级-系统架构设计师教程(清华第2版)【第15章 面向服务架构设计理论与实践(P527~554)-思维导图】​
  • # 利刃出鞘_Tomcat 核心原理解析(八)-- Tomcat 集群
  • #我与虚拟机的故事#连载20:周志明虚拟机第 3 版:到底值不值得买?
  • $.ajax()参数及用法
  • $L^p$ 调和函数恒为零
  • (vue)el-cascader级联选择器按勾选的顺序传值,摆脱层级约束
  • (超简单)使用vuepress搭建自己的博客并部署到github pages上
  • (剑指Offer)面试题34:丑数
  • (全部习题答案)研究生英语读写教程基础级教师用书PDF|| 研究生英语读写教程提高级教师用书PDF
  • (深度全面解析)ChatGPT的重大更新给创业者带来了哪些红利机会
  • (十七)Flink 容错机制
  • (算法设计与分析)第一章算法概述-习题
  • (原+转)Ubuntu16.04软件中心闪退及wifi消失
  • (转)Unity3DUnity3D在android下调试
  • (转)visual stdio 书签功能介绍
  • (转)项目管理杂谈-我所期望的新人
  • ... fatal error LINK1120:1个无法解析的外部命令 的解决办法
  • .net core 使用js,.net core 使用javascript,在.net core项目中怎么使用javascript
  • .NET MAUI学习笔记——2.构建第一个程序_初级篇
  • .net refrector
  • .Net6使用WebSocket与前端进行通信