刷穿剑指offer-Day15-哈希表II Python&Java的哈希表方法与解题套路!

简介: 刷穿剑指offer-Day15-哈希表II Python&Java的哈希表方法与解题套路!

昨日回顾


昨天我们开始了哈希表的学习,讲解了哈希表的集中实现方式。并通过一道 设计哈希集合 的题目,让我们将哈希表的理论转化为实践。

今天,我们就开始正式学习哈希表在Python与Java中的使用方式。在Java中,哈希表有两个数据类型 HashMap 与 HashSet,它们对应Python中的 dict 与 set ,下面我们开始分类学习!


HashSet & set


我们在昨天的设计哈希集合题目中,对HashSet已经有了一个初步的了解。HashSet与set 是一个无序不重复的元素集,集合在我们日常算法中对数组去重、已搜索的节点记录等有很大帮助。

具体方法如下:

操作 Python Java
初始化 s = set() HashSet<Integer> s = new HashSet<>();
增加 s.add(1) s.add(1)
删除 s.remove(1) 不存在报错
s.discard(2) 不存在不报错
s.pop()随机弹出并返回
s.remove(1) 不存在不报错
包含 1 in s s.contains(1)
获取长度 len(s) s.size()
清空 s.clear() s.clear()
查看是否为空 if not s s.isEmpty()

Python针对集合还可存在update更新、difference 并可以使用 | & 等方法,但在算法中的使用频度并不高,大家下来可以自行复习。


HashMap & dict


就如力扣 twoSum 这道题中,题目要求我们在数组中查找等于 target 的两个元素,并返回这两个数字的下标。如果只是判断能否找到这两个数,我们初始化一个HashSet,在遍历数组的过程中检查target - num 是否在存在HashSet中,如果找到直接返回True,否则将当前数字保存至HashSet中。

但既然要返回下标,就需要在保存数字的同时记录该数字的下标。使用方式如下图:


网络异常,图片无法展示
|

让我们来看看Python和Java中HashMap 和dict 都有哪些方法吧:

操作 Python Java
初始化 d = {} d = dict() HashMap<Integer,Integer> d = new HashMap<>();
增加键值对、已存在则修改 d[2] = 0 d.put(2,0);
获取key对应value d[2] d.get(2)
获取key,不存在则返回默认 d.get(3, -1) d.getOrDefault(3, -1)
删除某个键 del d[5] 不存在报错
d.pop(5, -1) 不存在返回默认值机
d.popitem() 随机删除一个键值对
d.remove(5) 不存在不报错
当key不存在时添加 d.setdefault(9, 0) d.putIfAbsent(9, 0)
获取长度 len(s) s.size()
包含 1 in s s.contains(1)
清空 s.clear() s.clear()
查看是否为空 if not s s.isEmpty()

剩余一些通用的d.keys() d.values() d.items()获取字典所有键、值、键值对等就不再赘述了,大家下来可以自行复习。

关于HashSet、set和HashMap、dict的相关知识就复习到这里。下来让我们做第一题熟悉熟悉?嗯...还是换个稍微有点难度的吧。


剑指OfferII032.有效的变位词


https://leetcode-cn.com/problems/dKk3P7/solution/shua-chuan-jian-zhi-offer-day15-ha-xi-bi-5pqx/

难度:简单


题目

给定两个字符串 s 和 t ,编写一个函数来判断它们是不是一组变位词(字母异位词)。

注意:若 s 和 t 中每个字符出现的次数都相同且字符顺序不完全相同,则称 s 和 t 互为变位词(字母异位词)。

进阶: 如果输入字符串包含 unicode 字符怎么办?你能否调整你的解法来应对这种情况?

提示:

  • 1 <= s.length, t.length <= 5 * 10 ^ 4
  • s and t 仅包含小写字母


示例

示例 1:
输入: s = "anagram", t = "nagaram"
输出: true
示例 2:
输入: s = "rat", t = "car"
输出: false
示例 3:
输入: s = "a", t = "a"
输出: false


分析

在不看进阶的情况下,这道题其实用不到哈希表,因为s和t只包含小写字母,我们在字符串和数组那章节已经讲过了这种题目的快速解法。

我们通过构造一个长度为26的数组对应26个英文字母(嗯...突然想起,谭警官的28个英文字母倒背如流,怕了怕了...)

通过字符串与ascii对应的方式完成匹配,这套路用的太多就不细讲了。

但是,进阶中说了,如果字符串包含unicode字符串怎么办?

怎么办,凉拌....直接上哈希表就行了啊!

其他操作和数据没什么区别,只是换成了哈希表的对应方法而已么。


数组解题


Python:

class Solution:
    def isAnagram(self, s: str, t: str) -> bool:
        if len(s) != len(t) or s == t:
            return False
        comp = [0] * 26
        for i in s:
            comp[ord(i) - 97] += 1
        for j in t:
            index = ord(j) - 97
            if comp[index] < 1:
                return False
            comp[index] -= 1
        return True


Python:

class Solution {
    public boolean isAnagram(String s, String t) {
        if (s.length() != t.length() || s.equals(t)) {
            return false;
        }
        int[] arr = new int[26];
        for (char i:s.toCharArray()) {
            arr[i - 97]++;
        }
        for (char j:t.toCharArray()) {
            arr[j - 97]--;
            if (arr[j - 97] < 0){
                return false;
            }
        }
        return true;
    }
}


哈希表解题


Python:

class Solution:
    def isAnagram(self, s: str, t: str) -> bool:
        return s != t and Counter(s) == Counter(t)


Python:

class Solution {
    public boolean isAnagram(String s, String t) {
        if (s.length() != t.length() || s.equals(t)) {
            return false;
        }
        HashMap<Character, Integer> arr = new HashMap<>();
        for (char i : s.toCharArray()) {
            arr.put(i, arr.getOrDefault(i, 0) + 1);
        }
        for (char j : t.toCharArray()) {
            if (!arr.containsKey(j) || arr.get(j) == 0)
                return false;
            arr.put(j, arr.get(j) - 1);
        }
        return true;
    }
}

不得不说Python做这种类型的题目简直是无脑啊....请恕我偷懒了,哈哈。

经过这道简单题的铺垫,再来看下面这道中等题,可以说就那会儿事儿....手速题而已!


剑指OfferII033.变位词组


https://leetcode-cn.com/problems/sfvd7V/solution/shua-chuan-jian-zhi-offer-day15-ha-xi-bi-p57n/

难度:中等


题目

给定一个字符串数组 strs ,将 变位词 组合在一起。 可以按任意顺序返回结果列表。

注意:若两个字符串中每个字符出现的次数都相同,则称它们互为变位词。

提示:

  • 1 <= strs.length <= 104
  • 0 <= strs[i].length <= 100
  • strs[i] 仅包含小写字母


示例

示例 1:
输入: strs = ["eat", "tea", "tan", "ate", "nat", "bat"]
输出: [["bat"],["nat","tan"],["ate","eat","tea"]]
示例 2:
输入: strs = [""]
输出: [[""]]
示例 3:
输入: strs = ["a"]
输出: [["a"]]


分析

这道题在难度方面并不高,思路其实剑指OfferII032类似的:

  1. 我们创建一个哈希表s,key 为String,value 为List
  2. 然后循环列表中的每个字符串,先将字符串排序
  3. 再看排序后的字符串是否在哈希表中,如果在则追加,如果不在单独开辟一对key:value即可
  4. 最终将哈希表的value值转化为列表返回即可。

但对于 Python 这里还有一个优化点的,如果按照上面的方式,我们需要创建一个哈希表嵌套列表的操作。

而且最终有需要将哈希表的value转化为列表再返回比较麻烦。

我们可以换个思路:

  1. 我们单独创建一个哈希表s和列表li
  2. 当哈希表中不存在排序后的字符串时,我们获取当前列表长度作为value(其实是列表的下标)
  3. 然后向哈希表中插入 key = 排序后的字符串, value = 该字符串在列表中的下标。
  4. 下次遇到同类型的字符串,我们直接在列表对应下标中插入该字符串即可。


解题


Python:

class Solution:
    def groupAnagrams(self, strs):
        ret = []
        d = {}
        for i in strs:
            sort_i = ''.join(sorted(i))
            if sort_i in d:
                ret[d[sort_i]].append(i)
            else:
                d[sort_i] = len(ret)
                ret.append([i])
        return ret


Java:

class Solution {
    public List<List<String>> groupAnagrams(String[] strs) {
        HashMap<String, ArrayList<String>> map = new HashMap<String, ArrayList<String>>();
        for (String str : strs) {
            char[] array = str.toCharArray();
            Arrays.sort(array);
            String key = new String(array);
            ArrayList<String> list = map.getOrDefault(key, new ArrayList<String>());
            list.add(str);
            map.put(key, list);
        }
        return new ArrayList<List<String>>(map.values());
    }
}

好了,今天的哈希表学习就到这里,题目虽然基础,但是一定要练习啊!




相关文章
|
26天前
|
消息中间件 Java Kafka
在Java中实现分布式事务的常用框架和方法
总之,选择合适的分布式事务框架和方法需要综合考虑业务需求、性能、复杂度等因素。不同的框架和方法都有其特点和适用场景,需要根据具体情况进行评估和选择。同时,随着技术的不断发展,分布式事务的解决方案也在不断更新和完善,以更好地满足业务的需求。你还可以进一步深入研究和了解这些框架和方法,以便在实际应用中更好地实现分布式事务管理。
|
25天前
|
机器学习/深度学习 Python
堆叠集成策略的原理、实现方法及Python应用。堆叠通过多层模型组合,先用不同基础模型生成预测,再用元学习器整合这些预测,提升模型性能
本文深入探讨了堆叠集成策略的原理、实现方法及Python应用。堆叠通过多层模型组合,先用不同基础模型生成预测,再用元学习器整合这些预测,提升模型性能。文章详细介绍了堆叠的实现步骤,包括数据准备、基础模型训练、新训练集构建及元学习器训练,并讨论了其优缺点。
43 3
|
1月前
|
Java
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
69 9
|
12天前
|
安全
Python-打印99乘法表的两种方法
本文详细介绍了两种实现99乘法表的方法:使用`while`循环和`for`循环。每种方法都包括了步骤解析、代码演示及优缺点分析。文章旨在帮助编程初学者理解和掌握循环结构的应用,内容通俗易懂,适合编程新手阅读。博主表示欢迎读者反馈,共同进步。
|
20天前
|
JSON 安全 API
Python调用API接口的方法
Python调用API接口的方法
86 5
|
29天前
|
算法 决策智能 Python
Python中解决TSP的方法
旅行商问题(TSP)是寻找最短路径,使旅行商能访问每个城市一次并返回起点的经典优化问题。本文介绍使用Python的`ortools`库解决TSP的方法,通过定义城市间的距离矩阵,调用库函数计算最优路径,并打印结果。此方法适用于小规模问题,对于大规模或特定需求,需深入了解算法原理及定制策略。
37 15
|
25天前
|
安全 Java 开发者
Java中WAIT和NOTIFY方法必须在同步块中调用的原因
在Java多线程编程中,`wait()`和`notify()`方法是实现线程间协作的关键。这两个方法必须在同步块或同步方法中调用,这一要求背后有着深刻的原因。本文将深入探讨为什么`wait()`和`notify()`方法必须在同步块中调用,以及这一机制如何确保线程安全和避免死锁。
37 4
|
25天前
|
Java
深入探讨Java中的中断机制:INTERRUPTED和ISINTERRUPTED方法详解
在Java多线程编程中,中断机制是协调线程行为的重要手段。了解和正确使用中断机制对于编写高效、可靠的并发程序至关重要。本文将深入探讨Java中的`Thread.interrupted()`和`Thread.isInterrupted()`方法的区别及其应用场景。
25 4
|
23天前
|
Java 数据处理 数据安全/隐私保护
Java处理数据接口方法
Java处理数据接口方法
25 1
|
27天前
|
机器学习/深度学习 人工智能 算法
强化学习在游戏AI中的应用,从基本原理、优势、应用场景到具体实现方法,以及Python在其中的作用
本文探讨了强化学习在游戏AI中的应用,从基本原理、优势、应用场景到具体实现方法,以及Python在其中的作用,通过案例分析展示了其潜力,并讨论了面临的挑战及未来发展趋势。强化学习正为游戏AI带来新的可能性。
72 4
下一篇
DataWorks