代码随想录-刷题第七天

news/2024/10/24 2:36:20/

454. 四数相加II

题目链接:454. 四数相加II

思路:哈希法。使用map集合,key存放a+b的值,value存放a+b出现的次数。使用两层循环,循环前两个数组,找出a+b,对map赋值。再用两层循环,遍历后两个数组,找出符合map中符合目标的值,并通过value获取出现的次数并累加。(其实就是将四数相加变成两数相加,将时间复杂度从O(n4)降至O(n2)

时间复杂度:O(n2)

class Solution {public int fourSumCount(int[] nums1, int[] nums2, int[] nums3, int[] nums4) {int count = 0;    // 统计a+b+c+d = 0 出现的次数Map<Integer, Integer> map = new HashMap<>();// 先遍历前两个数组,将a+b以及出现的次数放到map中for (int a : nums1) {for (int b : nums2) {map.put(a + b, map.getOrDefault(a + b, 0) + 1);}}// 然后遍历后两个数组,从map中找到符合条件的a+b并计数for (int c : nums3) {for (int d : nums4) {if (map.containsKey(0 - (c + d))) {count += map.get(0 - (c + d));}}}return count;}
}

383. 赎金信

题目链接:383. 赎金信

思路:哈希法。其实就是字母异位词的扩展题目,思路同字母异位词。

时间复杂度:O(n)

class Solution {public boolean canConstruct(String ransomNote, String magazine) {// 变向的字母异位词int[] count = new int[26];for (int i = 0; i < ransomNote.length(); i++) {count[ransomNote.charAt(i) - 'a']++;}for (int i = 0; i < magazine.length(); i++) {count[magazine.charAt(i) - 'a']--;}for (int i : count) {if (i > 0) {   // 仅在此处有差别return false;}}return true;}
}

15. 三数之和

题目链接:15. 三数之和

思路:使用双指针法。(本题的重点在于考察去重操作。)先对数组进行排序。使用i遍历一遍数组,遍历过程中,left初始为i+1,right初始为最后一个元素,然后如果left和right指向的元素符合目标值,将三个数放进结果中;如果不符合目标值,调整left和right的位置。

注意:要对三个数都进行去重操作。

i指向的是a,如果和前一个元素重复,就没必要再进行遍历了,跳过执行下一个元素。

left指向的是b,如果存入结果后,后一个元素仍然和当前元素相同,跳过后一个元素。

right指向的是c,如果存入结果后,前一个元素仍然和当前元素相同,跳过前一个元素。

通过双指针将时间复杂度由O(n3)降至了O(n2)

时间复杂度:O(n2)

class Solution {public List<List<Integer>> threeSum(int[] nums) {// 双指针法List<List<Integer>> res = new LinkedList<>();// 先进行排序Arrays.sort(nums);for (int i = 0; i < nums.length; i++) {// 如果当前最小的元素都大于0了,返回res就可以了if (nums[i] > 0) return res;// 对a去重,如果和前一个元素相同,说明已经判断过了,执行下一个if (i > 0 && nums[i] == nums[i - 1]) continue;int left = i + 1, right = nums.length - 1;while (left < right) {if (nums[i] + nums[left] + nums[right] < 0) {left++;} else if (nums[i] + nums[left] + nums[right] > 0) {right--;} else {res.add(Arrays.asList(nums[i], nums[left], nums[right]));while (left < right && nums[left] == nums[left + 1]) { // 对b去重left++;}while (left < right && nums[right] == nums[right - 1]) { // 对c去重right--;}left++;right--;}}}return res;}
}

8. 四数之和

题目链接:8. 四数之和

思路:使用双指针法,原理同三数之和。因为这次目标值可以指定为负数,所以要注意剪枝时的操作。用i和j确定a和b,用left和right寻找符合条件c和d。(同样要注意,这四个数,每个数都要进行去重操作。)

不要判断nums[k] > target 就返回了,三数之和 可以通过 nums[i] > 0 就返回了,因为 0 已经是确定的数了,四数之和这道题目 target是任意值。比如:数组是[-4, -3, -2, -1]target-10,不能因为-4 > -10而跳过。但是我们依旧可以去做剪枝,逻辑变成nums[i] > target && (nums[i] >= 0 || target >= 0)就可以了。

四数之和的双指针解法是两层for循环nums[k] + nums[i]为确定值,依然是循环内有left和right下标作为双指针,找出nums[k] + nums[i] + nums[left] + nums[right] == target的情况。

那么一样的道理,五数之和、六数之和等等都采用这种解法。

时间复杂度:O(n3)

class Solution {public List<List<Integer>> fourSum(int[] nums, int target) {// 双指针法,类似三数之和List<List<Integer>> res = new LinkedList<>();// 先排序Arrays.sort(nums);for (int i = 0; i < nums.length; i++) {if (nums[i] > target && (nums[i] >= 0 || target >= 0)){// 剪枝操作return res;}if (i > 0 && nums[i] == nums[i - 1]) { // 对a去重continue;}for (int j = i + 1; j < nums.length; j++) {if (j > i + 1 && nums[j] == nums[j - 1]) // 对b去重continue;int left = j + 1, right = nums.length - 1;while (left < right) {if (nums[i] + nums[j] + nums[left] + nums[right] < target) {left++;} else if (nums[i] + nums[j] + nums[left] + nums[right] > target) {right--;} else {res.add(Arrays.asList(nums[i], nums[j], nums[left], nums[right]));while (left < right && nums[left] == nums[left + 1]) {// 对c去重left++;}while (left < right && nums[right] == nums[right - 1]) {// 对d去重right--;}left++;right--;}}}}return res;}
}

哈希表题目总结

从哈希表的理论基础到数组、set和map的经典应用,把哈希表的全貌呈现出来。

同时也强调虽然map是万能的,详细介绍了什么时候用数组,什么时候用set

相信通过代码随想录中的哈希表总结篇,大家可以对哈希表有一个全面的了解。



http://www.ppmy.cn/news/1239271.html

相关文章

解决几乎任何机器学习问题 -- 学习笔记(组织机器学习项目)

书籍名&#xff1a;Approaching (Almost) Any Machine Learning Problem-解决几乎任何机器学习问题 此专栏记录学习过程&#xff0c;内容包含对这本书的翻译和理解过程 我们首先来看看文件的结构。对于你正在做的任何项目,都要创建一个新文件夹。在本例中,我 将项目命名为 “p…

排序....

1.直接插入排序 1.1将数组分为有序和无序部分 无序部分插入到有序部分4 #include<iostream> using namespace std; void InsertSort(int a[], int n) { int tmp; int j; for (int i 1; i < n; i) { if (a[i] < a[i - 1]) { tmp …

Tars框架 Tars-Go 学习

Tars 框架安装 网上安装教程比较多&#xff0c;官方可以参数这个 TARS官方文档 (tarsyun.com) 本文主要介绍部署应用。 安装完成后Tars 界面 增加应用amc 部署申请 amc.GoTestServer.GoTestObj 名称不知道的可以参考自己创建的app config 点击刷新可以看到自己部署的应用 服…

python pdf转txt文本、pdf转json

文章目录 一、前言二、实现方法1. 目录结构2. 代码 一、前言 此方法只能转文本格式的pdf&#xff0c;如果是图片格式的pdf需要用到ocr包&#xff0c;以后如果有这方面需求再加这个方法 二、实现方法 1. 目录结构 2. 代码 pdf2txt.py 代码如下 #!/usr/bin/env python # -*- …

nginx知识梳理及配置详解

软件开发全文档获取&#xff1a;点我获取 nginx安装 #nginx安装 yum -y install gcc pcre-devel openssl-devel #依赖包 useradd -s /sbin/nologin nginx ./configure --prefix/usr/local/nginx #指定安装目录 --usernginx #指定用户 --with-http_ss…

C++算法:字符串中的查找与替换

本周推荐阅读 C二分算法&#xff1a;得到子序列的最少操作次数 题目 你会得到一个字符串 s (索引从 0 开始)&#xff0c;你必须对它执行 k 个替换操作。替换操作以三个长度均为 k 的并行数组给出&#xff1a;indices, sources, targets。 要完成第 i 个替换操作: 检查 子字符…

获取当日的日期三个月后的日期

使用 java.time.LocalDate 类进行计算 import java.time.LocalDate;public class ThreeMonthsLaterExample {public static void main(String[] args) {// 获取当前日期LocalDate currentDate LocalDate.now();// 添加三个月LocalDate threeMonthsLater currentDate.plusMont…

excel单元格加背景颜色不生效?

如果在 Excel 中设置单元格背景颜色而发现不生效&#xff0c;可能有几个原因。以下是一些常见的解决方法&#xff1a; 1. **单元格锁定&#xff1a;** 检查所在单元格是否被锁定。如果单元格被锁定&#xff0c;并且工作表被保护&#xff0c;你可能无法更改其背景颜色。在工作表…