2023-04-14 算法面试中常见的查找表问题

news2025/1/27 19:38:36

2023-04-14 算法面试中常见的查找表问题

1 Set的使用

LeetCode349号问题:两个数组的交集

给定两个数组,编写一个函数来计算它们的交集。

示例 1:

输入: nums1 = [1,2,2,1], nums2 = [2,2]
输出: [2]
示例 2:

输入: nums1 = [4,9,5], nums2 = [9,4,9,8,4]
输出: [9,4]
说明:

输出结果中的每个元素一定是唯一的。
我们可以不考虑输出结果的顺序。
package Chapter04SetAndMap.IntersectionofTwoArrays;

import java.util.*;

/***********************************************************
 * @note      : 
 * @author    : l00379880 梁山广
 * @version   : V1.0 at 2019/8/20 15:46
 ***********************************************************/
class Solution {

    /**
     * 用retainAll方法实现求交集
     */
    public int[] intersection(int[] nums1, int[] nums2) {
        Set<Integer> numSet = new HashSet<>();
        for (int i : nums1) {
            numSet.add(i);
        }

        Set<Integer> num2Set = new HashSet<>();

        for (int i : nums2) {
            num2Set.add(i);
        }

        // 求交集
        numSet.retainAll(num2Set);
        int[] result = new int[numSet.size()];
        int i = 0;
        for (Integer num : numSet) {
            result[i] = num;
            i++;
        }
        return result;
    }


    /**
     * 用自己的方法实现交集,核心是contains方法
     */
    public int[] intersection2(int[] nums1, int[] nums2) {

        TreeSet<Integer> record = new TreeSet<>();
        for (int num : nums1) {
            record.add(num);
        }

        TreeSet<Integer> resultSet = new TreeSet<>();
        for (int num : nums2) {
            if (record.contains(num)) {
                resultSet.add(num);
            }
        }

        int[] res = new int[resultSet.size()];
        int index = 0;
        for (Integer num : resultSet) {
            res[index++] = num;
        }

        return res;
    }


    public static void main(String[] args) {
        int[] nums1 = {4, 9, 5};
        int[] nums2 = {9, 4, 9, 8, 4};
        int[] result = new Solution().intersection2(nums1, nums2);
        System.out.println(Arrays.toString(result));
    }
}

2 Map的使用

350.两个数组的交集 II

给定两个数组,编写一个函数来计算它们的交集。

示例 1:

输入: nums1 = [1,2,2,1], nums2 = [2,2]
输出: [2,2]
示例 2:

输入: nums1 = [4,9,5], nums2 = [9,4,9,8,4]
输出: [4,9]
说明:

输出结果中每个元素出现的次数,应与元素在两个数组中出现的次数一致。
我们可以不考虑输出结果的顺序。
进阶:

package Chapter04SetAndMap.IntersectionofTwoArrays2;

import java.util.*;

/***********************************************************
 * @note      : 
 * @author    : l00379880 梁山广
 * @version   : V1.0 at 2019/8/20 16:36
 ***********************************************************/
class Solution {
    public int[] intersect(int[] nums1, int[] nums2) {
        Map<Integer, Integer> num1Map = new HashMap<>();
        Set<Integer> num1Set = new HashSet<>();
        for (int i : nums1) {
            num1Map.merge(i, 1, Integer::sum);
            num1Set.add(i);
        }


        Map<Integer, Integer> num2Map = new HashMap<>();
        Set<Integer> num2Set = new HashSet<>();

        for (int i : nums2) {
            num2Map.merge(i, 1, Integer::sum);
            num2Set.add(i);
        }

        // 求交集
        num1Set.retainAll(num2Set);

        // num1Set就是交集的元素了
        List<Integer> resultList = new ArrayList<>();
        for (Integer num : num1Set) {
            int count = Math.min(num1Map.get(num), num2Map.get(num));
            for (int i = 0; i < count; i++) {
                resultList.add(num);
            }
        }

        int[] result = new int[resultList.size()];
        int i = 0;
        for (Integer num : resultList) {
            result[i] = num;
            i++;
        }
        return result;
    }

    public static void main(String[] args) {
        int[] nums1 = {1, 2, 2, 1};
        int[] nums2 = {2};
        int[] result = new Solution().intersect(nums1, nums2);
        System.out.println(Arrays.toString(result));
    }
}

3 Set和Map不同底层实现的区别

参考Part2Basic全课程,每部分实现地数组、链表和树可以用来作为Map的底层,不同的实现比较如下:
Set和Map不同底层实现的区别

哈希表虽然性能高,劣势是空间占用大,而且元素不能保持顺序~~

LeetCode上更多相似问题:242、202、290、205、451

242.有效的字母异位词

基于频率表的实现

class Solution {
    // 字母异位词:字母相同但是顺序不同的单词,,类似问题是438号问题
    public boolean isAnagram(String s, String t) {
        if("".equals(s) && "".equals(t)){
            return true;
        } 
        if("".equals(s) || "".equals(t) || s.length() != t.length()){
            return false;
        }
        int[] freq = new int[256];
        int len = s.length();
        for(int i = 0; i< len; i++){
            freq[s.charAt(i)] = freq[s.charAt(i)] + 1;
            freq[t.charAt(i)] = freq[t.charAt(i)] - 1;
        }
        for(int i = 0; i< len; i++){
            if(freq[s.charAt(i)] != 0){
                return false;
            }
        }
        return true;
    }
}

进一步优化,缩小数组范围,性能最高:

class Solution {
    // 字母异位词:字母相同但是顺序不同的单词,,类似问题是438号问题
    public boolean isAnagram(String s, String t) {
        if("".equals(s) && "".equals(t)){
            return true;
        } 
        if("".equals(s) || "".equals(t) || s.length() != t.length()){
            return false;
        }
        int[] freq = new int[26];
        int len = s.length();
        for(int i = 0; i< len; i++){
            int sInt = s.charAt(i) - 'a';
            int tInt = t.charAt(i) - 'a';
            freq[sInt] = freq[sInt] + 1;
            freq[tInt] = freq[tInt] - 1;
        }
        for(int i = 0; i< len; i++){
            if(freq[s.charAt(i) - 'a'] != 0){
                return false;
            }
        }
        return true;
    }
}

基于HashMap的实现

class Solution {
    /**
     * 字母异位词:字母相同但是顺序不同的单词,,类似问题是438号问题
     */
    public boolean isAnagram(String s, String t) {
        if ("".equals(s) && "".equals(t)) {
            return true;
        }
        if ("".equals(s) || "".equals(t) || s.length() != t.length()) {
            return false;
        }
        int len = s.length();
        // 哈希表,类似用来存频率
        Map<Character, Integer> cMap = new HashMap<>();
        for (int i = 0; i < len; i++) {
            char sC = s.charAt(i);
            char tC = t.charAt(i);
            if (cMap.get(sC) == null) {
                cMap.put(sC, 1);
            } else {
                cMap.put(sC, cMap.get(sC) + 1);
            }
            if (cMap.get(tC) == null) {
                cMap.put(tC, -1);
            } else {
                cMap.put(tC, cMap.get(tC) - 1);
            }
        }
        for (Character c : cMap.keySet()) {
            if (cMap.get(c) != 0) {
                return false;
            }
        }
        return true;
    }
}

202.快乐数

关键是观察规律

import java.util.HashSet;
import java.util.Set;

class Solution {
    int getSquareSum(int n) {
        int sum = 0;
        while (n != 0) {
            sum += (n % 10) * (n % 10);
            n = n / 10;
        }
        return sum;
    }

    /**
     * 若一个数是快乐数,最终变换会回到1,因此确定循环终结条件;
     * 若不是快乐数,会进入死循环,如何终至死循环,将每次变换过后的值存入HashSet中,判断是否出现过重复值,出现则return false;
     */
    public boolean isHappy(int n) {
        Set<Integer> set = new HashSet<>();
        while (true) {
            n = getSquareSum(n);
            if (n == 1) {
                return true;
            }
            if (set.contains(n)) {
                // 出现重复值了,直接退出
                return false;
            } else {
                set.add(n);
            }
        }
    }

    public static void main(String[] args) {
        // 19->82->100->1
        int n = 19;
        System.out.println(new Solution().isHappy(n));
    }
}

290.单词规律

class Solution {
    public boolean wordPattern(String pattern, String str) {
        if("".equals(pattern) || "".equals(str)){
            return false;
        }
        String[] words = str.split(" ");
        int len = pattern.length();
        if(len != words.length){
            return false;
        }
        Set<Character> setPattern = new HashSet<>();
        Set<String> setWord = new HashSet<>();
        for(int i = 0; i < len; i++){
            setPattern.add(pattern.charAt(i));
            setWord.add(words[i]);
        }
        if(setPattern.size() != setWord.size()){
            return false;
        }
        Map<Character, String> map = new HashMap<>();
        for(int i = 0; i < len; i++){
            Character c = pattern.charAt(i);
            String word = map.get(c);
            if(word == null){
                map.put(c, words[i]);
            }else {
                // 映射在之前就存在了,但是之前对应的关系和现在不符,直接返回false
                if(!word.equals(words[i])){
                    return false;
                }
            }
        }
        return true;
    }
}

205.同构字符串

和290几乎完全相同,单词边字符而已

import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;

class Solution {
    public boolean isIsomorphic(String s, String t) {
        if ("".equals(s) && "".equals(t)) {
            return true;
        }
        if ("".equals(s) || "".equals(t)) {
            return false;
        }
        int len = s.length();
        if (len != t.length()) {
            return false;
        }
        Set<Character> setS = new HashSet<>();
        Set<Character> setT = new HashSet<>();
        for (int i = 0; i < len; i++) {
            setS.add(s.charAt(i));
            setT.add(t.charAt(i));
        }
        if (setS.size() != setT.size()) {
            return false;
        }
        Map<Character, Character> map = new HashMap<>();
        for (int i = 0; i < len; i++) {
            Character cS = s.charAt(i);
            Character cSMap = map.get(cS);
            Character cT = t.charAt(i);
            if (cSMap == null) {
                map.put(cS, cT);
            } else {
                // 映射在之前就存在了,但是之前对应的关系和现在不符,直接返回false
                if (!cT.equals(cSMap)) {
                    return false;
                }
            }
        }
        return true;
    }

    public static void main(String[] args) {
        String s = "egg";
        String t = "add";
        System.out.println(new Solution().isIsomorphic(s, t));
    }
}

451.根据字符出现频率排序

class Solution {
    public String frequencySort(String s) {
        StringBuilder sb = new StringBuilder();
        int[] freq = new int[256];
        int len = s.length();
        for(int i = 0; i < len; i++){
            char c = s.charAt(i);
            freq[c] += 1;
        }

        for(int i = 0; i < len; i++){
            // max记录频率值最大的索引(字符的ASCII值,即在freq中的下标)
            int max = 0;
            for(int j = 0; j < 256; j++) {
                if(freq[j] > freq[max]){
                    max = j;
                }
            }
            // 插入频率个字符
            for(int m = 0; m < freq[max]; m++){
                sb.append((char)max);
            }
            // 一定记得清零
            freq[max] = 0;
        }
        return sb.toString();
    }

    public static void main(String[] args) {
        String s = "cccaaa";
        System.out.println(new Solution().frequencySort(s));
    }
}

4 Two Sum的问题

LeetCode第1号问题

定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标。

你可以假设每种输入只会对应一个答案。但是,你不能重复利用这个数组中同样的元素。

示例:

给定 nums = [2, 7, 11, 15], target = 9

因为 nums[0] + nums[1] = 2 + 7 = 9
所以返回 [0, 1]

第一种解法:把整个数组当做查找表

import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;

class Solution {
    public int[] twoSum(int[] nums, int target) {
        // 值和索引组成Map
        Map<Integer, Integer> mapValIndex = new HashMap<>();
        int len = nums.length;
        for (int i = 0; i < len; i++) {
            // 重复的元素不要二次加入
            mapValIndex.putIfAbsent(nums[i], i);
        }
        int[] result = new int[2];
        for (int i = 0; i < len; i++) {
            Integer index = mapValIndex.get(target - nums[i]);
            if (index != null && index != i) {
                // 注意顺序问题, 当nums=[3,3],target=6时会体现Math函数的作用
                result[0] = Math.min(i, index);
                result[1] = Math.max(i, index);
                // 找到就退出,因为用例保证了答案是唯一地了
                break;
            }
        }
        return result;
    }

    public static void main(String[] args) {
        int[] nums = {7, 3, 3};
        int target = 6;
        int[] result = new Solution().twoSum(nums, target);
        System.out.println(Arrays.toString(result));
    }
}

第2种解法:把遍历过程前面的元素作为查找表,顺便还能保证顺序
把遍历过程前面的元素作为查找表

class Solution {
      public int[] twoSum(int[] nums, int target) {
        // 值和索引组成Map
        Map<Integer, Integer> mapValIndex = new HashMap<>();
        int len = nums.length;
        for (int i = 0; i < len; i++) {
            Integer index = mapValIndex.get(target - nums[i]);
            if (index != null) {
                // 找到就退出,因为用例保证了答案是唯一地了
                return new int[]{index, i};
            }
            // 加入当前节点
            mapValIndex.put(nums[i], i);
        }
        return new int[0];
    }
}

相关的题目:

  • 15. 三数之和

    a+b = -c,转化为Two Sum的问题,但是注意这里的TwoSum可能不止有唯一解

    给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且不重复的三元组。
    注意:答案中不可以包含重复的三元组。
    例如, 给定数组 nums = [-1, 0, 1, 2, -1, -4],
    
    满足要求的三元组集合为:
    [
      [-1, 0, 1],
      [-1, -1, 2]
    ]
    
    /***********************************************************
     * @Description : LeetCode第15号问题:三数之和
     * https://leetcode-cn.com/problems/3sum/
     * @author      : 梁山广(Liang Shan Guang)
     * @date        : 2020/1/16 22:31
     * @email       : liangshanguang2@gmail.com
     ***********************************************************/
    package Chapter03.ThreeSum;
    
    import java.util.*;
    
    class Solution {
        /**
         * 这里的Two Sum和LeetCode第一题不同,这里一个target可能对应多个结果
         *
         * @param nums   要检索的数组
         * @param target 要求的目标值
         * @return 符合条件的下标对
         */
        public List<int[]> twoSum(int[] nums, int target) {
            List<int[]> resultList = new ArrayList<>();
            // 值和索引组成Map
            Map<Integer, Integer> mapValIndex = new HashMap<>();
            int len = nums.length;
            for (int i = 0; i < len; i++) {
                Integer index = mapValIndex.get(target - nums[i]);
                if (index != null) {
                    // 找到了一个满足条件地
                    resultList.add(new int[]{index, i});
                }
                // 加入当前节点
                mapValIndex.put(nums[i], i);
            }
            return resultList;
        }
    
        /**
         * 三个数字之和
         *
         * @param nums 数组
         * @return 所有满足三个数之和的元素对
         */
        public List<List<Integer>> threeSum(int[] nums) {
            // 这个排序很重要!!可能省掉下面大量的Collections.sort()
            Arrays.sort(nums);
            Set<List<Integer>> result = new HashSet<>();
            if (nums.length < 3) {
                return new ArrayList<>();
            }
            int len = nums.length;
            for (int i = 2; i < len; i++) {
                // 时间优化
                if (nums[i] == nums[i - 1] && nums[i] == nums[i - 2]) {
                    if (nums[i] == 0) {
                        // 把三个0加进去
                        result.add(new ArrayList<>(Arrays.asList(0, 0, 0)));
                    }
                    // 三个连续相等地,第3个开始就不用看了
                    continue;
                }
                // 注意twoSum返回地是nums的索引下标
                List<int[]> twoSumResultList = twoSum(Arrays.copyOfRange(nums, 0, i), -nums[i]);
                for (int[] twoSumResult : twoSumResultList) {
                    // 找到的两个索引不能和当前索引相等
                    if (twoSumResult.length == 2) {
                        // 这个是set,会自动去重
                        result.add(Arrays.asList(nums[twoSumResult[0]], nums[twoSumResult[1]], nums[i]));
                    }
                }
            }
            return new ArrayList<>(result);
        }
    
        /**
         * 输出顺序不重要,结果对就行~~
         * nums = {-1, 0, 1, 2, -1, -4}   ===> [ [-1, 0, 1], [-1, -1, 2]]
         * {-4, -2, -2, -2, 0, 1, 2, 2, 2, 3, 3, 4, 4, 6, 6}  ==> [[-4,-2,6],[-4,0,4],[-4,1,3],[-4,2,2],[-2,-2,4],[-2,0,2]]
         */
        public static void main(String[] args) {
            int[] nums = {-4, -2, -2, -2, 0, 1, 2, 2, 2, 3, 3, 4, 4, 6, 6};
            List<List<Integer>> lists = new Solution().threeSum(nums);
            System.out.println(lists);
        }
    }
    
  • 18. 四数之和
    给定一个包含 n 个整数的数组 nums 和一个目标值 target,判断 nums 中是否存在四个元素 a,b,c 和 d ,使得 a + b + c + d 的值与 target 相等?找出所有满足条件且不重复的四元组。
    
    注意:
    
    答案中不可以包含重复的四元组。
    
    示例:
    给定数组 nums = [1, 0, -1, 0, -2, 2],和 target = 0。
    
    满足要求的四元组集合为:
    [
      [-1,  0, 0, 1],
      [-2, -1, 1, 2],
      [-2,  0, 0, 2]
    ]
    
    /***********************************************************
     * @Description : LeetCode第18号问题:四数之和
     * https://leetcode-cn.com/problems/4sum/
     * @author      : 梁山广(Liang Shan Guang)
     * @date        : 2020/1/16 23:46
     * @email       : liangshanguang2@gmail.com
     ***********************************************************/
    package Chapter03.FourSum;
    
    import java.util.*;
    
    class Solution {
        /**
         * 这里的Two Sum和LeetCode第一题不同,这里一个target可能对应多个结果
         *
         * @param nums   要检索的数组
         * @param target 要求的目标值
         * @return 符合条件的元素列表
         */
        public List<List<Integer>> twoSum(int[] nums, int target) {
            List<List<Integer>> resultList = new ArrayList<>();
            // 值和索引组成Map
            Map<Integer, Integer> mapValIndex = new HashMap<>();
            int len = nums.length;
            for (int i = 0; i < len; i++) {
                Integer index = mapValIndex.get(target - nums[i]);
                if (index != null) {
                    // 找到了一个满足条件地
                    resultList.add(new ArrayList<>(Arrays.asList(nums[index], nums[i])));
                }
                // 加入当前节点
                mapValIndex.put(nums[i], i);
            }
            return resultList;
        }
    
        /**
         * 三个数字之和
         *
         * @param nums 数组
         * @return 所有满足三个数之和的元素对
         */
        public List<List<Integer>> threeSum(int[] nums, int target) {
            // 这个排序很重要!!可能省掉下面大量的Collections.sort()
            Arrays.sort(nums);
            Set<List<Integer>> result = new HashSet<>();
            if (nums.length < 3) {
                return new ArrayList<>();
            }
            int len = nums.length;
            for (int i = 2; i < len; i++) {
                // 时间优化
                if (nums[i] == nums[i - 1] && nums[i] == nums[i - 2]) {
                    if (nums[i] + nums[i - 1] + nums[i - 2] == target) {
                        // 把三个加进去
                        result.add(new ArrayList<>(Arrays.asList(nums[i], nums[i - 1], nums[i - 2])));
                    }
                    // 三个连续相等地,第3个开始就不用看了
                    continue;
                }
                // 注意twoSum返回地是nums的索引下标
                List<List<Integer>> twoSumResultList = twoSum(Arrays.copyOfRange(nums, 0, i), target - nums[i]);
                for (List<Integer> twoSumResult : twoSumResultList) {
                    // 找到的两个索引不能和当前索引相等
                    if (twoSumResult.size() == 2) {
                        // 这个是set,会自动去重
                        result.add(Arrays.asList(twoSumResult.get(0), twoSumResult.get(1), nums[i]));
                    }
                }
            }
            return new ArrayList<>(result);
        }
    
    
        public List<List<Integer>> fourSum(int[] nums, int target) {
            // 这个排序很重要!!可能省掉下面大量的Collections.sort()
            Arrays.sort(nums);
            Set<List<Integer>> result = new HashSet<>();
            if (nums.length < 4) {
                return new ArrayList<>();
            }
            int len = nums.length;
            for (int i = 3; i < len; i++) {
                // 时间优化
                if (nums[i] == nums[i - 1] && nums[i] == nums[i - 2] && nums[i] == nums[i - 3]) {
                    if (nums[i] + nums[i - 1] + nums[i - 2] + nums[i - 3] == target) {
                        // 把三个加进去
                        result.add(new ArrayList<>(Arrays.asList(nums[i], nums[i - 1], nums[i - 2], nums[i - 3])));
                    }
                    // 四个连续相等地,第4个开始就不用看了
                    continue;
                }
                // 注意twoSum返回地是nums的索引下标
                List<List<Integer>> threeSumResultList = threeSum(Arrays.copyOfRange(nums, 0, i), target - nums[i]);
                for (List<Integer> threeSumResult : threeSumResultList) {
                    // 找到的两个索引不能和当前索引相等
                    if (threeSumResult.size() == 3) {
                        // 这个是set,会自动去重
                        result.add(Arrays.asList(threeSumResult.get(0), threeSumResult.get(1), threeSumResult.get(2), nums[i]));
                    }
                }
            }
            return new ArrayList<>(result);
        }
    
        /**
         * 输出顺序不重要,结果对就行~~
         * nums = {-1, 0, 1, 2, -1, -4}   ===> [ [-1, 0, 1], [-1, -1, 2]]
         * {-4, -2, -2, -2, 0, 1, 2, 2, 2, 3, 3, 4, 4, 6, 6}  ==> [[-4,-2,6],[-4,0,4],[-4,1,3],[-4,2,2],[-2,-2,4],[-2,0,2]]
         */
        public static void main(String[] args) {
            int[] nums = {0, 0, 0, 0};
            int target=1;
            List<List<Integer>> lists = new Solution().fourSum(nums, target);
            System.out.println(lists);
        }
    }
    
  • 16.最接近的三数之和
    给定一个包括 n 个整数的数组 nums 和 一个目标值 target。找出 nums 中的三个整数,使得它们的和与 target 最接近。返回这三个数的和。假定每组输入只存在唯一答案。
    例如,给定数组 nums = [-121-4], 和 target = 1.
    
    与 target 最接近的三个数的和为 2. (-1 + 2 + 1 = 2).
    
    /***********************************************************
     * @Description : LeetCode第16号问题:最接近的三数之和
     * https://leetcode-cn.com/problems/3sum-closest/
     * @author      : 梁山广(Liang Shan Guang)
     * @date        : 2020/1/17 00:19
     * @email       : liangshanguang2@gmail.com
     ***********************************************************/
    package Chapter03.ThreeSumClosest;
    
    import java.util.*;
    
    class Solution {
    
        /**
         * 这里的Two Sum和LeetCode第一题不同,这里一个target可能对应多个结果
         *
         * @param nums   要检索的数组
         * @param target 要求的目标值
         * @return 符合条件的元素列表
         */
        public List<List<Integer>> twoSum(int[] nums, int target) {
            List<List<Integer>> resultList = new ArrayList<>();
            // 值和索引组成Map
            Map<Integer, Integer> mapValIndex = new HashMap<>();
            int len = nums.length;
            for (int i = 0; i < len; i++) {
                Integer index = mapValIndex.get(target - nums[i]);
                if (index != null) {
                    // 找到了一个满足条件地
                    resultList.add(new ArrayList<>(Arrays.asList(nums[index], nums[i])));
                }
                // 加入当前节点
                mapValIndex.put(nums[i], i);
            }
            return resultList;
        }
    
        /**
         * 三个数字之和
         *
         * @param nums 数组
         * @return 所有满足三个数之和的元素对
         */
        public List<List<Integer>> threeSum(int[] nums, int target) {
            // 这个排序很重要!!可能省掉下面大量的Collections.sort()
            Arrays.sort(nums);
            Set<List<Integer>> result = new HashSet<>();
            if (nums.length < 3) {
                return new ArrayList<>();
            }
            int len = nums.length;
            for (int i = 2; i < len; i++) {
                // 时间优化
                if (nums[i] == nums[i - 1] && nums[i] == nums[i - 2]) {
                    if (nums[i] + nums[i - 1] + nums[i - 2] == target) {
                        // 把三个加进去
                        result.add(new ArrayList<>(Arrays.asList(nums[i], nums[i - 1], nums[i - 2])));
                    }
                    // 三个连续相等地,第3个开始就不用看了
                    continue;
                }
                // 注意twoSum返回地是nums的索引下标
                List<List<Integer>> twoSumResultList = twoSum(Arrays.copyOfRange(nums, 0, i), target - nums[i]);
                for (List<Integer> twoSumResult : twoSumResultList) {
                    // 找到的两个索引不能和当前索引相等
                    if (twoSumResult.size() == 2) {
                        // 这个是set,会自动去重
                        result.add(Arrays.asList(twoSumResult.get(0), twoSumResult.get(1), nums[i]));
                    }
                }
            }
            return new ArrayList<>(result);
        }
    
        public int threeSumClosest(int[] nums, int target) {
            int len = nums.length;
            if (len < 3) {
                return -1;
            }
            if (len == 3) {
                return nums[0] + nums[1] + nums[2];
            }
            Arrays.sort(nums);
            int minTarget = nums[0] + nums[1] + nums[2];
            int maxTarget = nums[len - 3] + nums[len - 2] + nums[len - 1];
            if (target >= maxTarget) {
                return maxTarget;
            }
            if (target <= minTarget) {
                return minTarget;
            }
            // 临时的targetTmp和target的距离,从最小开始找,只要找到一个就退出,即是最小的distance
            int distance = 0;
            int maxDistance = Math.max(maxTarget - target, target - minTarget);
            // target一定在minTarget和maxTarget之间了
            while (distance < maxDistance) {
                if (target + distance <= maxTarget && threeSum(nums, target + distance).size() != 0) {
                    return target + distance;
                }
                if (target - distance >= minTarget && threeSum(nums, target - distance).size() != 0) {
                    return target - distance;
                }
                distance++;
            }
            return -1;
        }
    
        public static void main(String[] args) {
            int[] nums = {-1, 2, 1, -4};
            int target = 1;
            System.out.println(new Solution().threeSumClosest(nums, target));
        }
    }
    

5 灵活选择键值 454号问题 4Sum

解题思路:把C+D的组合放入查找表中,通过查找A+B是否等于-(C+D)

给定四个包含整数的数组列表 A , B , C , D ,计算有多少个元组 (i, j, k, l) ,使得 A[i] + B[j] + C[k] + D[l] = 0。

为了使问题简单化,所有的 A, B, C, D 具有相同的长度 N,且 0N500 。所有整数的范围在 -2^282^28 - 1 之间,最终结果不会超过 2^31 - 1 。
例如:

输入:
A = [ 1, 2]
B = [-2,-1]
C = [-1, 2]
D = [ 0, 2]

输出:
2

解释:
两个元组如下:
1. (0, 0, 0, 1) -> A[0] + B[0] + C[0] + D[1] = 1 + (-2) + (-1) + 2 = 0
2. (1, 1, 0, 0) -> A[1] + B[1] + C[0] + D[0] = 2 + (-1) + (-1) + 0 = 0
/***********************************************************
 * @Description : 给定四个包含整数的数组列表 A , B , C , D ,计算有多少个元组 (i, j, k, l) ,使得 A[i] + B[j] + C[k] + D[l] = 0
 * 解题思路:把C+D的组合放入查找表中,通过查找A+B是否等于-(C+D)
 * @author      : 梁山广(Liang Shan Guang)
 * @date        : 2019/8/20 23:52
 * @email       : liangshanguang2@gmail.com
 ***********************************************************/
package Chapter04.TwoSum2;

import java.util.HashMap;
import java.util.Map;

/*
 *时间复杂度是:O(n^2)
 *空间复杂度是:O(n^2)
 */
class Solution {
    public int fourSumCount(int[] A, int[] B, int[] C, int[] D) {
        Map<Integer, Integer> record = new HashMap<>();
        for (int i = 0; i < C.length; i++) {
            for (int j = 0; j < D.length; j++) {
                if (record.get(C[i] + D[j]) == null) {
                    record.put(C[i] + D[j], 1);
                } else {
                    record.put(C[i] + D[j], record.get(C[i] + D[j]) + 1);
                }
            }
        }
        int result = 0;
        for (int i = 0; i < A.length; i++) {
            for (int j = 0; j < B.length; j++) {
                if (record.containsKey(-A[i] - B[j])) {
                    result += record.get(-A[i] - B[j]);
                }
            }
        }
        return result;
    }

    public static void main(String[] args) {
        int[] A = {1, 2};
        int[] B = {-2, -1};
        int[] C = {-1, 2};
        int[] D = {0, 2};
        int result = new Solution().fourSumCount(A, B, C, D);
        System.out.println(result);
    }
}

类似问题:49. 字母异位词分组

下面的实现行用时 :13 ms, 在所有 Java 提交中击败了66.68%的用户;内存消耗 :40.9 MB, 在所有 Java 提交中击败了96.06%的用户

核心优化:数组或者字符串长度不要单独赋值

// 教训:len不要单独赋值
class Solution {
    public List<List<String>> groupAnagrams(String[] strs) {
        List<List<String>> result = new ArrayList<>();
        Map<String, List<String>> anagramsMap = new HashMap<>();
        for (int i = 0; i < strs.length; i++) {
            char[] chs = strs[i].toCharArray();
            Arrays.sort(chs);
            String strSorted = new String(chs);
            if (anagramsMap.get(strSorted)==null){
                List<String> anagrams = new ArrayList<>();
                anagrams.add(strs[i]);
                anagramsMap.put(strSorted, anagrams);
            }else {
                List<String> anagrams = anagramsMap.get(strSorted);
                anagrams.add(strs[i]);
                anagramsMap.put(strSorted,anagrams);
            }
        }

        for (String key : anagramsMap.keySet()) {
            result.add(anagramsMap.get(key));
        }
        return result;
    }
}

4.6 灵活选择键值:LeetCode447号问题-回旋镖的数量

选择距离作为键

class Solution {
    /**
     * 获取距离,暂时不开方,只要能表示距离的唯一值即可
     */
    int getDistance(int[] p1, int[] p2) {
        return (p2[0] - p1[0]) * (p2[0] - p1[0]) + (p2[1] - p1[1]) * (p2[1] - p1[1]);
    }

    public int numberOfBoomerangs(int[][] points) {
        int count = 0;
        for (int i = 0; i < points.length; i++) {
            // 到i的距离和相等距离点的个数组成的映射
            Map<Integer, Integer> mapDistancePoints = new HashMap<>();
            for (int j = 0; j < points.length; j++) {
                int distance = getDistance(points[i], points[j]);
                if (mapDistancePoints.get(distance) == null) {
                    mapDistancePoints.put(distance, 1);
                } else {
                    mapDistancePoints.put(distance, mapDistancePoints.get(distance) + 1);
                }
            }
            // 统计到points[i]距离相等的点数
            for (Integer distance : mapDistancePoints.keySet()) {
                int cnt = mapDistancePoints.get(distance);
                // 指定的举例值至少有一个元素地,所以这里不需要判空
                if (cnt > 1) {
                    // 因为考虑顺序,所以求组合而不是排列A-N-2 = N * (N - 1)
                    count += cnt * (cnt - 1);
                }
            }
        }
        return count;
    }
}

类似问题:LeetCode第149号问题:直线上最多的点数

坑非常多,同时锻炼价值也很大,用例考察地角度非常多。执行用时 :14 ms, 在所有 Java 提交中击败了81.61%的用户;内存消耗 :36 MB, 在所有 Java 提交中击败了60.83%的用户

import java.util.HashMap;
import java.util.Map;

class Solution {

    public int maxPoints(int[][] points) {
        int max = 0;
        if (points.length < 3) {
            return points.length;
        }
        for (int i = 0; i < points.length; i++) {
            // 到i的斜率和斜率相等的点的个数组成的映射
            Map<Double, Integer> mapKPoints = new HashMap<>();
            // 和points[i]的横纵坐标都相同的点,初始至少有自己
            int sameCount = 1;
            for (int j = i + 1; j < points.length; j++) {
                // 斜率可能是浮点数
                double k = Integer.MAX_VALUE;
                if (points[i][1] != points[j][1]) {
                    k = (points[i][0] - points[j][0]) * 1.0 / (points[i][1] - points[j][1]);
                } else {
                    if (points[i][0] == points[j][0]) {
                        // 横纵坐标都相等,那么这样的点可以加到任何一条和i相连的直线上
                        sameCount++;
                        continue;
                    }
                }
                // 下面这一行是为了解决用例[[4,0],[4,-1],[4,5]] ,Java中0.0和-0.0是不同的,https://blog.csdn.net/ouy5933/article/details/72461239
                k += 0.0;
                if (mapKPoints.get(k) == null) {
                    mapKPoints.put(k, 1);
                } else {
                    mapKPoints.put(k, mapKPoints.get(k) + 1);
                }
            }
            // 统计到points[i]距离相等的点数
            for (Double k : mapKPoints.keySet()) {
                int cnt = mapKPoints.get(k);
                // 要算上当前点points[i],所以要cnt+1
                max = Math.max(cnt + sameCount, max);
            }
            // 全部都是一个点的用例:[[1,1],[1,1],[1,1]]
            max = Math.max(sameCount, max);
        }
        return max;
    }

    /***
     * 典型用例:
     *
     * {{1, 1}, {2, 2}, {3, 3}}  ==>3
     * {{0, 0}}   ==> 1
     * {{0, 0},{0, 0}}  ==> 2
     * {{0, 0},{1, 1},{0, 0}}  ==> 3
     * [[1,1],[1,1],[2,2],[2,2]]  ==> 4
     *  {{-4, 1}, {-7, 7}, {-1, 5}, {9, -25}}  ==》 3  这里例子终点考察斜率是浮点类型
     *  [[4,0],[4,-1],[4,5]]  ==>  3  考察 Java中0.0和-0.0是不同的
     */
    public static void main(String[] args) {
        int[][] points = {{4, 0}, {4, -1}, {4, 5}};
        System.out.println(new Solution().maxPoints(points));
    }
}

4.7 查找表+滑动数组:219号问题,Contains Duplicate II

使用Set巧妙控制滑动窗口

给定一个整数数组和一个整数 k,判断数组中是否存在两个不同的索引 i 和 j,使得 nums [i] = nums [j],并且 i 和 j 的差的绝对值最大为 k。
示例 1:
  输入: nums = [1,2,3,1], k = 3
  输出: true

示例 2:
  输入: nums = [1,0,1,1], k = 1
  输出: true
示例 3:
  输入: nums = [1,2,3,1,2,3], k = 2
  输出: false
/***********************************************************
 * @Description : 给定一个整数数组和一个整数 k,判断数组中是否存在两个不同的索引 i 和 j,使得 nums [i] = nums [j],并且 i 和 j 的差的绝对值最大为 k
 * 思路:控制一个宽度为不大于k的窗口(record)不断滑动,检测到有相等的两个不同下标的就返回true
 * @author      : 梁山广(Liang Shan Guang)
 * @date        : 2019/8/21 08:16
 * @email       : liangshanguang2@gmail.com
 ***********************************************************/
package Chapter04.ContainDuplicate;

import java.util.HashSet;
import java.util.Set;

/**
 * 时间复杂度O(n),空间复杂度O(k)
 */
public class Solution {
    public boolean containsNearbyDuplicate(int[] nums, int k) {
        Set<Integer> record = new HashSet<>();
        for (int i = 0; i < nums.length; i++) {
            if (record.contains(nums[i])) {
                // 前面加入的元素nums[j]包含已经包含nums[i]了,
                return true;
            }
            record.add(nums[i]);
            // 保持record中不超过k个元素,即i和j的索引最大差值不大于k,超过k时抹掉最左边的元素即可
            if (record.size() == k + 1) {
                record.remove(nums[i - k]);
            }
        }
        // 遍历到最后还找不到就返回false
        return false;
    }

    public static void main(String[] args) {
        int[] nums = {1, 2, 3, 1};
        int k = 3;
        boolean contain = new Solution().containsNearbyDuplicate(nums, k);
        System.out.println(contain);
    }
}

相似的问题:217.存在重复元素,相似的思路,只是不用维护窗口了

class Solution {
    public boolean containsDuplicate(int[] nums) {
        Set<Integer> set = new HashSet();
        for(int num : nums){
            if(set.contains(num)){
                return true;
            }
            set.add(num);
        }
        return false;
    }
}

4.8 二分搜索树底层实现的顺序性 220号问题 Contain Duplicate III

给定一个整数数组,判断数组中是否有两个不同的索引 i 和 j,使得 nums [i] 和 nums [j] 的差的绝对值最大为 t,并且 i 和 j 之间的差的绝对值最大为 ķ。

示例 1:
输入: nums = [1,2,3,1], k = 3, t = 0
输出: true
示例 2:

输入: nums = [1,0,1,1], k = 1, t = 2
输出: true
示例 3:

输入: nums = [1,5,9,1,5,9], k = 2, t = 3
输出: false

在上一节的基础上加一个判断即可:在大小为k的窗口中找值差不大于t的即可

220号问题1
220号问题2

/***********************************************************
 * @Description : 给定一个整数数组,判断数组中是否有两个不同的索引 i 和 j,使得 nums [i] 和 nums [j] 的差的绝对值最大为 t,并且 i 和 j 之间的差的绝对值最大为 ķ。

 * @author      : 梁山广(Liang Shan Guang)
 * @date        : 2019/8/21 08:16
 * @email       : liangshanguang2@gmail.com
 ***********************************************************/
package Chapter04.ContainDuplicate;

import java.util.TreeSet;

/**
 * 时间复杂度O(n),空间复杂度O(k)
 */
public class Solution {
    public boolean containsNearbyAlmostDuplicate(int[] nums, int k, int t) {
        TreeSet<Integer> record = new TreeSet<>();
        for (int i = 0; i < nums.length; i++) {
            // 在长度为k的窗口内存在一个下标为j的元素nums[j],使得abs(nums[i] - nums[j]) < t,推导出 nums[i]-t <= nums[j] <= nums[i] + t
            // 这里的nums[j]就取最接近窗口内大于"nums[i]-t"的最小值,如果这个值也小于"nums[i]+t",说明j满足题干条件,返回true即可
            if (record.ceiling(nums[i] - t) != null && record.ceiling(nums[i] - t) <= nums[i] +  t){
                // 前面加入的元素包含nums[k了]
                return true;
            }
            record.add(nums[i]);
            // 保持record中不超过k个元素,即i和j的索引最大差值不大于k,超过k时抹掉最左边的元素即可
            if (record.size() == k + 1) {
                record.remove(nums[i - k]);
            }
        }
        // 遍历到最后还找不到就返回false
        return false;
    }

    public static void main(String[] args) {
        int[] nums = {1, 2, 3, 1};
        int k = 3;
        int t = 0;
        boolean contain = new Solution().containsNearbyAlmostDuplicate(nums, k, t);
        System.out.println(contain);

    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/443688.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

直播预告 | 时序数据处理的云端利器:TDengine Cloud 详解与演示

当下&#xff0c;我们正处在一个万物互联的时代&#xff0c;大数据、云原生、AI、5G 等数字技术极大地方便了人们的生活&#xff0c;但智能物联网产生的海量数据却成为众多企业在数据处理上的巨大痛点。从本质来看&#xff0c;这些数据大多是产生自各种设备和传感器的时序数据&…

康耐视Designer,通过VC5与西门子S7-1200 PLCProfinet通讯详细

测试使用软件版本 Designer Version: 2.7 GSD: GSDML-V2.3-Cognex-VC5-20140828STEP 7 Professional V14Network Configurator Version: 3.56测试使用硬件 Cognex Vision Controller VC5CIC-5000Siemens PLC: S7-1200 CPU 1214C DC/DC/RLY1.PLC端设置 1.新建一个项目,添加对应…

没看错!一行python代码就可以帮您获取图片中的文字信息

最近工作中有需求需要用python对图片中的文字进行识别&#xff0c;调研了一下&#xff0c;选择了tesseract&#xff0c; 目前在github上有50.5k个star&#xff01;python可以调用&#xff0c;安装也十分方便&#xff0c;pip install pytesseract 即可。如果没有Pillow 包&…

ERTEC200P-2 PROFINET设备完全开发手册(7-2)

7.2 添加自定义参数 编辑GSDML文件&#xff0c;对于64 bytes DI 进行如下修改&#xff0c;由于一般的IO模块不需要支持PROFIEnergy&#xff0c;去掉PROFIEnergy支持&#xff1b;修改模块ID(ModuleIdentNumber "0x00000080"&#xff09;&#xff1b;子模块ID不变&am…

Linux中的进程优先级

Linux中的进程优先级 目录 Linux中的进程优先级引入1、基本概念2、查看进程3、PRI和NI4、修改进程优先级5、进程切换One more thing 引入 在Linux操作系统中&#xff0c;进程的优先级对于系统的性能和响应时间至关重要。进程的优先级决定了该进程在系统资源分配中所占的比例。…

说PHP不适合做爬虫的人,看这里

文章目录 一、关于PHP爬虫框架—Goutte1.1 什么是Goutte1.2 Goutte的优点1.3 Goutte的安装 二、Goutte的使用2.1 基本用法2.2 获取页面内容2.3 表单提交2.4 AJAX请求2.5 登录并抓取数据 三、15个Goutte爬虫示例3.1 简单示例3.2 获取表格内容3.3 登录并获取数据3.4 处理JavaScri…

遇到Spring事务失效,你该怎么办?

Spring 事务场景失效是一个常见的问题。今天来分析这个问题。 1、事务方法被final、static关键字修饰&#xff0c;方法访问权限不是public Service public class UserService {Autowiredprivate UserDao userDao;// final修饰的事务方法Transactionalpublic final void addUse…

实时分布式低延迟OLAP数据库Apache Pinot探索实操

文章目录 概述定义特性何时使用 部署Local安装快速启动手动设置集群 Docker安装快速启动手动启动集群Docker Compose 实操批导入数据流式导入数据 概述 定义 Apache Pinot 官网地址 https://pinot.apache.org/ 最新版本0.12.1 Apache Pinot 官网文档地址 https://docs.pinot.a…

Vivado如何清理工程,并避免缺失必要的文件?

FPGA开发是一项非常复杂的工作&#xff0c;需要处理大量的代码、约束和仿真文件。在开发过程中&#xff0c;由于文件数量庞大&#xff0c;很容易导致工程变得混乱不堪。Vivado是Xilinx公司推出的一款FPGA开发工具&#xff0c;能够帮助开发人员处理这种复杂性&#xff0c;提高开…

【FPGA-DSP】第七期:Mcode调用

目录 1. 有限状态机设计 1.1 FSM代码编写 1.2 输入序列代码编写 1.3. mcode概念 2. Simulink 2.1. 输入部分 2.2. Mcode模块 2.3. 完整模块 总结 参考博客【R1】&#xff0c;参考视频资料【R2】 设计中经常用到一些控制逻辑&#xff0c;如有限状态机&#xff08;FS…

语音特征参数MFCC提取过程详解

一、MFCC概述 在语音识别&#xff08;SpeechRecognition&#xff09;和话者识别&#xff08;SpeakerRecognition&#xff09;方面&#xff0c;最常用到的语音特征就是梅尔倒谱系数&#xff08;Mel-scaleFrequency Cepstral Coefficients&#xff0c;简称MFCC&#xff09;。根据…

IntelliJ IDEA全新发布v2023.1——全面增强UI体验和性能!

IntelliJ IDEA&#xff0c;是java编程语言开发的集成环境。IntelliJ在业界被公认为最好的java开发工具&#xff0c;尤其在智能代码助手、代码自动提示、重构、JavaEE支持、各类版本工具(git、svn等)、JUnit、CVS整合、代码分析、 创新的GUI设计等方面的功能可以说是超常的。 I…

javaSwingrsa和aes混合加密文件传输系统

RSA和AES加密混合文件传输系统 环境要求&#xff1a; 1、java1.8 2、socket 3、mysql 4、io流 5、swing 登录界面 客户端登录&#xff1a; 1.用户注册的账号保存到数据库中&#xff0c;用户的密码经过md5后保存到数据库中。 2.用户输入正确的账号和密码可以进入 “ …

Gulp 打包压缩 js 文件到指定目录详细流程(修改文件名与后缀)

主篇 前端 Gulp 详细介绍与案例使用。 打包 index.js 测试内容 js 包内容文件&#xff1a; // 测试属性 export const dzmName dzm export const dzmAge 18// 测试方法 export function dzmFunc (name, age) {return 名字&#xff1a;${name}&#xff0c;年龄&#xff1a;${…

Chat的提问工程师,要上线了你慌不慌?

一、什么是Prompt? (1)Prompt的概念 我们在使用GPT的时候,给GPT发送的消息就是Prompt. 例如,当我们问ChatGPT"WPS是什么软件?"时: 其中,"WPS是什么软件?"这句话就是Prompt. (2)为什么要学习Prompt? Prompt是我们和GPT交流的方式,我们需要让GPT了解…

微信小程序从零开始经验贴(含详细资料及链接)

背景: 从事微信小程序开发近3年&#xff0c;现在几乎能独当一面&#xff0c;对后台程序也有一定研究。      我学习的基础是c编程经验&#xff0c;对前端、后端没有太多涉及。微信小程序相关资料都是自己找的&#xff0c;然后在项目中看别人的代码&#xff0c;摸着石头过河…

什么是单例模式?

文章目录 01 | 什么是单例模式&#xff1f;02 | 实现懒汉式饿汉式 03 | 总结 每一次实验课都要把上一节课的实验报告打印出来&#xff0c;交作业&#xff0c;这个时候的打印店总是很多人&#xff0c;这时候打印机就那么几台&#xff0c;粥少僧多的情况下&#xff0c;打印机怎么…

动态路由的原理与配置

一.动态路由介绍 1.动态路由选择 指路由器使用路由选择协议来获悉网络并更新路由选择表。 2.路由协议分类 3.管理距离介绍 管理距离&#xff08;AD&#xff09;用于判断从邻接路由器收到的路由选择信息的可信度&#xff0c;它是 0-255的整数&#xff0c;0 表示可信度最大&#…

Android9 Settings源码导入Android Studio查看并调试

Android9 Settings源码导入Android Studio查看并调试 本次调试在已编译的工程下 1. 导入packages/app/settings 不要直接导入这个模块&#xff0c;导入上一级app目录 在选择时选择Settings相关所有目录&#xff0c;如下 如下这里的sdk配置是我已经添加的&#xff0c;后面可…

用了这几个方法,Facebook广告转化真上去了!

随着全球电商市值的猛涨&#xff0c;Facebook广告的价值再次被放大。然而&#xff0c;在Facebook上的广告资源竞争日益激烈的情况下&#xff0c;即使你有最好的广告策略和投放预算&#xff0c;如果你的广告创意不能吸引潜在客户的注意&#xff0c;那么你的广告投放也可能失败。…