[leetCode 丶 20241106] 1865. 找出和为指定值的下标对
原题:
https://leetcode.cn/problems/finding-pairs-with-a-certain-sum/description/
描述:
给你两个整数数组 nums1
和 nums2
,请你实现一个支持下述两类查询的数据结构:
1.累加 ,将一个正整数加到 nums2
中指定下标对应元素上。
2.计数 ,统计满足 nums1[i] + nums2[j]
等于指定值的下标对 (i, j)
数目(0 <= i < nums1.length 且 0 <= j < nums2.length
)。
实现 FindSumPairs
类:
FindSumPairs(int[] nums1, int[] nums2)
使用整数数组nums1
和nums2
初始化FindSumPairs
对象。void add(int index, int val)
将val
加到nums2[index]
上,即,执行nums2[index] += val
。int count(int tot)
返回满足nums1[i] + nums2[j] == tot
的下标对(i, j)
数目。
示例 1:
输入:
["FindSumPairs", "count", "add", "count", "count", "add", "add", "count"]
[[[1, 1, 2, 2, 2, 3], [1, 4, 5, 2, 5, 4]], [7], [3, 2], [8], [4], [0, 1], [1, 1], [7]]
输出:
[null, 8, null, 2, 1, null, null, 11]解释:
FindSumPairs findSumPairs = new FindSumPairs([1, 1, 2, 2, 2, 3], [1, 4, 5, 2, 5, 4]);
findSumPairs.count(7); // 返回 8 ; 下标对 (2,2), (3,2), (4,2), (2,4), (3,4), (4,4) 满足 2 + 5 = 7 ,下标对 (> 5,1), (5,5) 满足 3 + 4 = 7
findSumPairs.add(3, 2); // 此时 nums2 = [1,4,5,4,5,4]
findSumPairs.count(8); // 返回 2 ;下标对 (5,2), (5,4) 满足 3 + 5 = 8
findSumPairs.count(4); // 返回 1 ;下标对 (5,0) 满足 3 + 1 = 4
findSumPairs.add(0, 1); // 此时 nums2 = [2,4,5,4,5,4]
findSumPairs.add(1, 1); // 此时 nums2 = [2,5,5,4,5,4]
findSumPairs.count(7); // 返回 11 ;下标对 (2,1), (2,2), (2,4), (3,1), (3,2), (3,4), (4,1), (4,2), (4,4) 满足 2 + 5 = 7 ,下标对 (5,3), (5,5) 满足 3 + 4 = 7
提示
- 1 <= nums1.length <= 1000
- 1 <= nums2.length <= 10^5
- 1 <= nums1[i] <= 10^9
- 1 <= nums2[i] <= 10^5
- 0 <= index < nums2.length
- 1 <= val <= 10^5
- 1 <= tot <= 10^9
最多调用 add 和 count 函数各 1000 次
个人版答案
执行用时: 705 ms 执行内存消耗: 102.68 M
class FindSumPairs { Map<Integer, List<Integer>> nums1Indexs = new HashMap<>(); Map<Integer, List<Integer>> nums2Indexs = new HashMap<>(); int[] n1, n2; public FindSumPairs(int[] nums1, int[] nums2) { n1 = nums1; n2 = nums2; for(int i = 0; i < nums1.length; i++){ List<Integer> indexs1 = nums1Indexs.getOrDefault(nums1[i], new ArrayList<>()); indexs1.add(i); nums1Indexs.put(nums1[i], indexs1); } for(int i = 0; i < nums2.length; i++){ List<Integer> indexs2 = nums2Indexs.getOrDefault(nums2[i], new ArrayList<>()); indexs2.add(i); nums2Indexs.put(nums2[i], indexs2); } } public void add(int index, int val) { int tmp = n2[index]; // must List<Integer> indexs2 = nums2Indexs.getOrDefault(tmp, new ArrayList<>()); indexs2.removeIf(x -> x == index); if (indexs2.isEmpty()){ nums2Indexs.remove(tmp); }else { nums2Indexs.put(tmp, indexs2); } // add int newTmp = tmp + val; // swap n2[index] = newTmp; // may not List<Integer> new21 = nums2Indexs.getOrDefault(newTmp, new ArrayList<>()); new21.add(index); nums2Indexs.put(newTmp, new21); } public int count(int tot) { int c = 0; for(int i = 0; i < n1.length; i++){ c += nums2Indexs.getOrDefault(tot - n1[i], new ArrayList<>()).size(); } return c; } } /** * Your FindSumPairs object will be instantiated and called as such: * FindSumPairs obj = new FindSumPairs(nums1, nums2); * obj.add(index,val); * int param_2 = obj.count(tot); */
优秀解法
class FindSumPairs { private int[] nums1; private int[] nums2; private Map<Integer, Integer> map1; private Map<Integer, Integer> map2; public FindSumPairs(int[] nums1, int[] nums2) { this.nums1 = nums1; this.nums2 = nums2; this.map1 = new HashMap<>(); this.map2 = new HashMap<>(); for(int num : this.nums1){ this.map1.merge(num, 1, Integer::sum); } for(int num : this.nums2){ this.map2.merge(num, 1, Integer::sum); } } public void add(int index, int val) { this.map2.merge(this.nums2[index], -1, Integer::sum); this.nums2[index] += val; this.map2.merge(this.nums2[index], 1, Integer::sum); } public int count(int tot) { int result = 0; for(int num : this.nums1){ if(num >= tot) continue; result += this.map2.getOrDefault(tot - num, 0); } return result; } } /** * Your FindSumPairs object will be instantiated and called as such: * FindSumPairs obj = new FindSumPairs(nums1, nums2); * obj.add(index,val); * int param_2 = obj.count(tot); */
个人解题思路与优秀答案解析
题目分析及个人版思路
- 计算两个数组所有元素可组成的和的下标对的个数, 那我们易知遍历数组对所有数字求和即可
- 分析题目提示, 只有数组
nums2
会被求和操作, 那我们容易获得思路, 对nums2
所有相同元素, 统计下标, 则当求和时候我们使用和-nums1[i]
的值去获取之前储存的元素下标数组长度, 求和即可得下标对数量. - 注意题中所需要实现的方法, 当调用add的时候, 操作nums2后, 要重新统计相同元素的下标, 防止求和异常
- 详见代码.
进阶版思路
这把我败在了api上...哈哈哈哈哈哈, 我真没用过 map.merge 啊! 这下学到了. 思路是一致的, 看他代码就好
每日一姥
大佬
牛牛牛
好好好
膜拜,月月教我写代码
每日姨姥
大佬
听不懂听不懂
大佬