代码随想录算法训练营(JAVA) | 第三章 哈希表part01 DAY05

发布于:2024-02-27 ⋅ 阅读:(73) ⋅ 点赞:(0)

   今日任务 

力扣242. 有效的字母异位词349. 两个数组的交集202. 快乐数1. 两数之和

什么时候想到用哈希法,当我们遇到了要快速判断一个元素是否出现集合里的时候,就要考虑哈希法。

 什么时候使用哈希法,当我们需要查询一个元素是否出现过,或者一个元素是否在集合里的时候,就要第一时间想到哈希法。

使用数组和set来做哈希法的局限。

  • 数组的大小是受限制的,而且如果元素很少,而哈希值太大会造成内存空间的浪费。
  • set是一个集合,里面放的元素只能是一个key,而两数之和这道题目,不仅要判断y是否存在而且还要记录y的下标位置,因为要返回x 和 y的下标。所以set 也不能用。

242. 有效的字母异位词

思路

由于字符串只包含 26个小写字母,因此我们可以维护一个长度为 26 的频次数组 table,先遍历记录字符串 s 中字符出现的频次,然后遍历字符串 t,减去 table 中对应的频次

题解
class Solution {
    public boolean isAnagram(String s, String t) {
        if (s.length() != t.length()) {
            return false;
        }

        int[] hashTable = new int[26];
        for (int i = 0; i < s.length(); i ++ ) {
            hashTable[s.charAt(i) - 'a']++;
        }

        for (int i = 0; i < t.length(); i ++ ) {
            hashTable[t.charAt(i) - 'a']--;
            if (hashTable[t.charAt(i) - 'a'] < 0) {
                return false;
            }
        }
        return true;
    }
}

349. 两个数组的交集

思路

苦于 基础不牢语法不熟现去看了看基础依照思路写出来了

老是忘条件处理

题解
1.使用HashSet
class Solution {
    public int[] intersection(int[] nums1, int[] nums2) {
        if (nums1 == null || nums1.length == 0 || nums2 == null || nums2.length == 0) {
            return new int[0];
        }
        Set<Integer> set1 = new HashSet<>();
        Set<Integer> setList = new HashSet<>();

        for (int i : nums1) {
            set1.add(i);
        }

        for (int i : nums2) {
            if(set1.contains(i)) {
                setList.add(i);
            }
        }

        // 方法一: 将结果集合转化为数组
        // return setList.stream().mapToInt(x -> x).toArray();

        // 方法二: 创建一个新的数组存放setList中的值并返回
        int[] arr = new int[setList.size()];
        int flag = 0;
        for (int i : setList) {
            arr[flag++] = i;
        }
        return arr;
    }
}
2.使用哈希数组
class Solution {
    public int[] intersection(int[] nums1, int[] nums2) {
        int[] hash1 = new int[1010];
        int[] hash2 = new int[1010];

        List<Integer> resList = new ArrayList<>();

        for (int i : nums1) {
            hash1[i] ++;
        }

        for (int i : nums2) {
            hash2[i]++;
        }

        for (int i = 0; i < 1010; i ++) {
            if (hash1[i] > 0 && hash2[i] > 0) {
                resList.add(i);
            }
        }

        int index = 0;

        int[] res = new int[resList.size()];
        for (int i : resList) {
            res[index++] = i;
        }
        return res;
        
    }
}

202. 快乐数

思路

第一遍读题 emmmm没想法,看一眼提示   /思考  要存 set 也就是说这道题有重复的内容咯,每位数的平方和会重复!那也就是说会陷入循环,也就是说只要有重复内容就不是快乐数

题解
class Solution {
    public boolean isHappy(int n) {
        Set<Integer> set = new HashSet<>();
        while (n != 1 && !set.contains(n)) {
            set.add(n);
            n = getNext(n);
        }
        if (n == 1) {
            return true;
        }
        return false;
    }

    private int getNext(int n) {
        int sum = 0;
        while (n > 0) {
            int tmp = n % 10;
            sum += tmp * tmp;
            n /= 10;
        }
        return sum;
    }
}

1. 两数之和

思路

正如提示说的要用map KV,为什么用map呢?因为 要存 数值  和 下标 这两个值

题解 
class Solution {
    public int[] twoSum(int[] nums, int target) {
        int[] res = new int[2];

        Map<Integer, Integer> map = new HashMap<>();

        if (nums == null || nums.length == 0) {
            return res;
        }

        for (int i = 0; i < nums.length; i++ ) {
            int tmp = target - nums[i];
            if (map.containsKey(tmp)) {
                res[0] = map.get(tmp);
                res[1] = i;
                break;
            }
            map.put(nums[i], i);
        }
        return res;
    }
}

本文含有隐藏内容,请 开通VIP 后查看