算法02哈希法(完结)

1.哈希法理论基础

1.1哈希表

(1)哈希表

哈希表是一种数据结构,用于存储键值对(key-value pairs)。它通过将键(key)通过哈希函数映射到一个特定的索引位置来实现快速的数据访问。这个索引位置在内存中的数组或桶(buckets)中,使得在常数时间复杂度内可以进行查找、插入和删除操作。

想象一下你的家里有一个带有标签的抽屉。每个标签都对应着一个抽屉里的物品。当你需要某样东西时,你不必搜索整个房子,而是直接根据标签找到对应的抽屉,这就像哈希表根据键找到对应的数值一样。这种快速定位的方式使得你能够在瞬间找到你需要的物品,就像哈希表可以在常数时间内找到相应的值。

(2)哈希函数

哈希函数是一种将输入数据映射为固定长度散列值(哈希值)的函数。其主要目的是将任意长度的数据转换为固定长度的输出,通常是一个固定大小的数字或字节序列。

哈希函数具有以下特性:

  • 确定性: 相同的输入始终产生相同的哈希值。
  • 高效性: 计算速度快,能在合理时间内完成计算。
  • 离散性: 输入数据的微小变化应该导致输出哈希值的显著变化。
  • 不可逆性: 理论上不可通过哈希值逆向计算出原始输入数据。

常见的哈希函数有MD5、SHA-1、SHA-256等,它们被广泛用于数据加密、数据完整性验证、密码存储等领域。

想象你是一位魔术师,你有一个魔法箱子用来存放各种物品。你的目标是将每样物品放进箱子里,并在箱子的每个格子上放置一个标签。这个标签不仅告诉你物品存放在哪里,还得保证这个标签是独一无二的。你使用一个特殊的变化魔法(哈希函数),这个魔法会将每件物品都转化成一个独特的魔法标签,让你可以快速地找到它们。所以,当你需要取出某样物品时,你只需使用这个特殊魔法,它会让你知道这个物品的魔法标签,而这个标签对应着箱子的一个格子。这就好像哈希函数把数据变成一个特殊的“标签”,让你可以迅速找到存放的位置。而哈希函数的“魔法”在于,无论你放进去什么样的物品,它总是能给你一个独一无二的标签,就像每件物品都有一个特殊的魔法标签一样。

(3)哈希碰撞

在这里插入图片描述

哈希碰撞指的是不同的输入数据经过哈希函数计算后得到了相同的哈希值。在理想情况下,哈希函数应该能够将不同的输入映射到不同的哈希值,但在实际应用中,由于哈希函数将无限的输入空间映射到有限的输出空间,发生哈希碰撞是可能的。

想象一下你是一个魔术师,你的“蓝条”是有限的,当你的蓝条不足时,你的魔术可能会失灵而不准确。在你的魔法失效时,这就可能会发生原来是一个标签对应一个物品的情况而编程一个标签对应两个或两个以上的物品。“蓝条”就相当于是哈希表的存储空间,一个标签对应多个物品就是哈希碰撞

哈希冲突可以通过以下几种方法解决:

  • 开放寻址法(Open Addressing):这种方法在哈希冲突发生时,会寻找哈希表中的下一个可用位置,并尝试将数据存储在那里。这包括线性探测、二次探测、双重哈希等技术,逐个检查直到找到空槽来解决冲突。

在这里插入图片描述

  • 链表法(Chaining):哈希表中的每个槽位不只是一个单独的位置,而是一个链表或其他数据结构。当发生哈希冲突时,将新的键值对添加到该位置的链表中。这样,相同哈希值的元素都可以存储在同一个位置上,而不会发生覆盖。

在这里插入图片描述

  • 再哈希(Rehashing):当哈希表负载因子过高时,可以重新调整哈希表的大小,通常是增大容量,然后重新哈希所有的键值对到新的表中。这可以减少冲突的发生,因为新的更大的表提供了更多的空间来均匀分布键值对。

  • 完美哈希函数(Perfect Hashing):这是一种在特定情况下能够完全避免冲突的方法。完美哈希函数能够保证每个键都映射到不同的位置,但在实际中找到完美哈希函数可能比较困难。

选择哪种方法取决于应用的需求和数据特性。链表法在处理冲突时比较灵活,但需要更多的存储空间。开放寻址法则在空间效率上更高,但可能需要更多的探测步骤来解决冲突。再哈希和完美哈希函数则更多地关注于降低冲突的概率。

1.2哈希法基本思想

哈希法是一种基于哈希函数和哈希表的技术,用于将数据映射到一个固定范围的索引位置,以实现快速的查找、插入和删除操作。这个技术的核心是哈希函数,它将数据转换为哈希值,然后将该哈希值映射到哈希表中的特定位置。

1.3哈希法适用场景与最常用的哈希结构

在算法问题中,哈希法通常用于:

  • 快速查找: 哈希函数将数据映射为索引,使得在哈希表中能够以常数时间复杂度(O(1))进行查找操作。
  • 判断元素是否存在: 通过哈希表的结构,可以快速判断一个元素是否在集合中。
  • 去重操作: 将数据存储在哈希表中,可以自动去除重复元素,只保留唯一的元素。

2.LeetCode242:有效的字母异位词

给定两个字符串 s 和 t ,编写一个函数来判断 t 是否是 s 的字母异位词。

注意:若 s 和 t 中每个字符出现的次数都相同,则称 s 和 t 互为字母异位词。

示例 1:

输入: s = “anagram”, t = “nagaram” 输出: true

示例 2:

输入: s = “rat”, t = “car” 输出: false

提示:

1 <= s.length, t.length <= 5 * 104 s 和 t 仅包含小写字母

(1)图解本题的哈希内核

在这里插入图片描述

(2)cpp代码

//在s中出现的一个字母,我们就增加其在OrccrenceWord中的值
//在t中出现该字母,我们就减少其在orccrenceWord中的值
//如果s和t字符串是有效字母的异位词,OrccurenceWord的每一项最后应该都是0
//因为对一组异位词,s对一个字母提供的正增量刚好等于t对一个字母提供的负增量
class Solution {
public:
    bool isAnagram(string s, string t) {
        int OrccrenceWord[26] = {0};

        for(int i: s)
        {
            OrccrenceWord[i - 'a']++;
        }

        for(int i: t)
        {
            OrccrenceWord[i - 'a']--;
        }

        for(int i = 0; i < 26; i++)
        {
            if(OrccrenceWord[i] != 0)
            {
                return false;
            }
        }

        return true;

    }
};

3.LeetCode349:两个数组的交集

给定两个数组 nums1 和 nums2 ,返回 它们的交集 。输出结果中的每个元素一定是 唯一 的。我们可以 不考虑输出结果的顺序 。

示例 1:

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

示例 2:

输入:nums1 = [4,9,5], nums2 = [9,4,9,8,4] 输出:[9,4] 解释:[4,9] 也是可通过的

提示:

1 <= nums1.length, nums2.length <= 1000 0 <= nums1[i], nums2[i] <=
1000

(1)图解本题哈希内核

在这里插入图片描述

(2)cpp代码

//unordered_set是一种常用的数据结构,适合在原数据规模很大或者原数据十分离散的情况
//unordered_set就像我们数学中的集合一样,满足两个主要特性:1.无需;2.不重复
//result存储结果
//nums1_set利用这个数据结构(类)的构造函数,哈希映射nums1,对齐进行去重
//遍历nums2,如果nums2中的元素在nums1中出现了,就把它插入到结果哈希表(result)中,最后返回结果哈希表
class Solution {
public:
    vector<int> intersection(vector<int>& nums1, vector<int>& nums2) {
    
    unordered_set<int> result;
    unordered_set<int> nums1_set(nums1.begin(), nums1.end());

    for(int n: nums2)
    {
        if(nums1_set.find(n) != nums1_set.end())
        {
            result.insert(n);
        }
    }
    return vector<int>(result.begin(), result.end()); 

    }
};

4.LeetCode202:快乐数

编写一个算法来判断一个数 n 是不是快乐数。

「快乐数」 定义为:

对于一个正整数,每一次将该数替换为它每个位置上的数字的平方和。 然后重复这个过程直到这个数变为 1,也可能是 无限循环 但始终变不到 1。
如果这个过程 结果为 1,那么这个数就是快乐数。 如果 n 是 快乐数 就返回 true ;不是,则返回 false 。

(1)图解本题哈希内核:

在这里插入图片描述

(2)cpp代码:

class Solution {
public:

    int GetSum(int i)
    {
        int sum = 0;
        while(i)
        {
            sum += (i %10) * (i % 10);
            i /= 10;

        }
        return sum;
    }


    bool isHappy(int n) {
        std::unordered_set<int> set;

        while(1)
        {
            int sum = GetSum(n);
            auto iter = set.find(sum);
            
            if(iter != set.end() && *iter !=1)
            {
                return false;
            }
            else if (iter != set.end() && *iter == 1)
            {
                return true;
            }
            else{
            set.insert(sum);
            n = sum;
            }
            
        }

    }


};

5.LeetCode1:1. 两数之和

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

你可以假设每种输入只会对应一个答案。但是,数组中同一个元素在答案里不能重复出现。

你可以按任意顺序返回答案。

示例 1:

输入:nums = [2,7,11,15], target = 9 输出:[0,1] 解释:因为 nums[0] + nums[1] ==
9 ,返回 [0, 1] 。 示例 2:

输入:nums = [3,2,4], target = 6 输出:[1,2]

示例 3:

输入:nums = [3,3], target = 6 输出:[0,1]

只会存在一个有效答案

(1)图解本题哈希内核

在这里插入图片描述

(2)cpp代码

//用空间换时间,嵌套的两层for循环变成了不嵌套
//第一次遍历nums,将其映射到map这种键值对的数据结构中,哈希函数就是insert函数
//第二次遍历nums,用于检测是否存在满足target的数组元素
class Solution {
public:
    vector<int> twoSum(vector<int>& nums, int target) {
        std::unordered_map<int, int> map;
        for(int i = 0; i < nums.size(); i++)
        {
            map.insert(pair<int, int>(nums[i], i));
        }

        for(int i = 0; i < nums.size(); i++)
        {
            auto iter = map.find(target - nums[i]);
            if(iter != map.end() && iter->second != i)
            {
                return std::vector<int> {iter->second,i};
            }

        }
        return {};
        
    }
};

力扣上的大神更为简洁的代码:

//这种方式也是以空间换时间,但相较于我的代码,合并了for循环,稍稍节约了内存,因为不用把nums全部映射到map中再开始寻找target
class Solution {
public:
    vector<int> twoSum(vector<int>& nums, int target) {
        unordered_map<int, int> map;

        for(int i = 0; i <= nums.size(); i++)
        {
            auto iter = map.find(target - nums[i]);
            if(iter != map.end())
            {
                return {iter -> second, i};
            }
            map.insert(pair<int, int>(nums[i], i));
        }
        return {};
    }
};

6.LeetCode454:四数相加

给你四个整数数组 nums1、nums2、nums3 和 nums4 ,数组长度都是 n ,请你计算有多少个元组 (i, j, k, l) 能满足:

0 <= i, j, k, l < n nums1[i] + nums2[j] + nums3[k] + nums4[l] == 0

示例 1:

输入:nums1 = [1,2], nums2 = [-2,-1], nums3 = [-1,2], nums4 = [0,2] 输出:2

解释:

两个元组如下:

  1. (0, 0, 0, 1) -> nums1[0] + nums2[0] + nums3[0] + nums4[1] = 1 + (-2) + (-1) + 2 = 0
  2. (1, 1, 0, 0) -> nums1[1] + nums2[1] + nums3[0] + nums4[0] = 2 + (-1) + (-1) + 0 = 0

示例 2:

输入:nums1 = [0], nums2 = [0], nums3 = [0], nums4 = [0] 输出:1

1.暴力解法(四层for循环)

这种解法是可以解决此问题的,但是时间超时,只可以通过20多个测试用例

class Solution {
public:
    int fourSumCount(vector<int>& nums1, vector<int>& nums2, vector<int>& nums3, vector<int>& nums4) {
        int count = 0;
        for(int i = 0; i < nums1.size(); i++)
            for(int j = 0; j < nums2.size(); j++)
                for(int k = 0; k < nums3.size(); k++)
                    for(int m = 0; m < nums4.size(); m++)
                    {
                        if(nums1[i] + nums2[j] + nums3[k] + nums4[m] == 0)
                        count++;
                    }
    return count;

    }
};

2.优化后的暴力解法(三层for循环)

这种解法比上一种解法的时间复杂度稍低,可以通过50多个测试用例,但仍然超时

class Solution {
public:
    int fourSumCount(vector<int>& nums1, vector<int>& nums2, vector<int>& nums3, vector<int>& nums4) {
        int count = 0;
        std::unordered_map<int, int> map;
        for(int n: nums4)
        {
            map[n]++;
        }


        for(int i = 0; i < nums1.size(); i++)
            for(int j = 0; j < nums2.size(); j++)
                for(int k = 0; k < nums3.size(); k++){
                    int target = -(nums1[i] + nums2[j] + nums3[k]);
                    if(map.find(target) != map.end())
                        count += map[target]; 

                }

                    
    return count;

    }
};

3.继续优化(两层for循环)

class Solution {
public:
    int fourSumCount(vector<int>& nums1, vector<int>& nums2, vector<int>& nums3, vector<int>& nums4) {
        int count = 0;
        std::unordered_map<int, int> map;
        for(int i = 0; i < nums4.size(); i++)
            for(int j = 0; j < nums3.size(); j++)
                {
                    map[nums3[j] + nums4[i]] ++;
                }


        for(int i = 0; i < nums1.size(); i++)
            for(int j = 0; j < nums2.size(); j++)
               {
                    int target = -(nums1[i] + nums2[j]);
                    if(map.find(target) != map.end())
                        count += map[target]; 

                }

                    
    return count;

    }
};

4.图解本题思考步骤与哈希内核

在这里插入图片描述

7.一些思考

(1)关于解题和算法的使用

在解一道算法题时,我们可能会遇到两种情况:

  • 情况一:我们对这道题目类似的题目有相当多的经验并且一见到这到题就”本能“地想起这些经验,可以将经验引入这道题目中,或者说把这道题目归类为一类题型,进而使用一套方法论去解决这道题目
  • 情况二:我们对这道题目感到"陌生",最直接的解法就是暴力解法,在暴力完成之后,想到一些数据结构和算法,想到一些对时间复杂度和空间复杂度的优化方法,对暴力解法进行优化升级,以达到比较理想的解题效果

前者需要大量刷题,大量总结和积累,而后者要求在有一定的语言基础和算法与数据结构基础之上,有灵敏的思维。前者固然有方法论,但仍然需要"因地制宜",后者固然灵活,但也不是毫无章法可循。

(2)哈希法总结:

我们想要在我们的代码/算法中,快速判断一个元素是否在一个集合中出现,就可以考虑是否引用哈希法,而哈希法通常是基于一些数据结构实现的,下面是三种常见的数据据结构:

  • 数组
  • set(集合)
  • map(映射)

a)对set的进行进一步解析和总结

关于set,C++ 给提供了如下三种可用的数据结构:

  • std::set
  • std::multiset
  • std::unordered_set

在这里插入图片描述

std::set是一个有序的集合容器,它存储不重复的元素。内部实现通常是基于红黑树(一种自平衡的二叉查找树)。它的元素按照特定的排序顺序排列,默认情况下是升序。插入、删除和查找操作的时间复杂度是对数级别的。它保持元素的唯一性,不允许重复的元素存在。

示例:

#include <iostream>
#include <set>

int main() {
    std::set<int> mySet = {3, 1, 4, 1, 5, 9};

    for (const auto& elem : mySet) {
        std::cout << elem << " ";
    }
    // 输出结果将是:1 3 4 5 9

    return 0;
}

std::multiset: std::multiset 也是一个有序的集合容器,与 std::set类似,但允许存储重复的元素。它内部同样基于红黑树实现。与 std::set不同的是,它允许多个元素拥有相同的值,且在插入、删除和查找时都保持元素的插入顺序。

示例:

#include <iostream>
#include <set>

int main() {
    std::multiset<int> myMultiSet = {3, 1, 4, 1, 5, 9};

    for (const auto& elem : myMultiSet) {
        std::cout << elem << " ";
    }
    // 输出结果将是:1 1 3 4 5 9

    return 0;
}

std::unordered_set是一个无序的集合容器,它使用哈希表作为内部实现。哈希表使得插入、删除和查找操作的时间复杂度接近于常数级别,但不保持元素的顺序。它存储不重复的元素,不允许重复的值存在。

示例:

#include <iostream>
#include <unordered_set>

int main() {
    std::unordered_set<int> myUnorderedSet = {3, 1, 4, 1, 5, 9};

    for (const auto& elem : myUnorderedSet) {
        std::cout << elem << " ";
    }
    // 输出结果可能是:9 5 4 3 1 (无序)

    return 0;
}

b)对map的进一步总结和归纳

C++提供如下三种map:

  • std::map
  • std::multimap
  • std::unordered_map

std::map是有序的关联容器,存储键值对,并根据键的顺序进行排序。它内部通常使用红黑树实现,确保插入、删除和查找操作的平均时间复杂度为对数级别。

示例:

#include <iostream>
#include <map>

int main() {
    std::map<int, std::string> myMap;
    myMap[1] = "One";
    myMap[3] = "Three";
    myMap[2] = "Two";

    for (const auto& pair : myMap) {
        std::cout << pair.first << ": " << pair.second << std::endl;
    }

    return 0;
}

std::multimap 也是有序的关联容器,与 std::map 类似,但允许存储多个具有相同键的元素。它内部同样使用红黑树实现。

示例:

#include <iostream>
#include <map>

int main() {
    std::multimap<int, std::string> myMultiMap;
    myMultiMap.insert({1, "One"});
    myMultiMap.insert({3, "Three"});
    myMultiMap.insert({2, "Two"});
    myMultiMap.insert({1, "Another One"});

    for (const auto& pair : myMultiMap) {
        std::cout << pair.first << ": " << pair.second << std::endl;
    }

    return 0;
}

std::unordered_map 是无序的关联容器,它使用哈希表作为内部实现,提供了更快的查找速度(平均情况下为常数时间)。然而,由于无序性,它不保持元素的顺序。

示例:

#include <iostream>
#include <unordered_map>

int main() {
    std::unordered_map<int, std::string> myUnorderedMap;
    myUnorderedMap[1] = "One";
    myUnorderedMap[3] = "Three";
    myUnorderedMap[2] = "Two";

    for (const auto& pair : myUnorderedMap) {
        std::cout << pair.first << ": " << pair.second << std::endl;
    }

    return 0;
}

这三种映射都提供了不同的特性和适用场景。std::map 和 std::multimap 适用于需要有序映射的场景,而 std::unordered_map 则适用于需要高效查找,但不需要保持元素顺序的场景。

c)数组、set、map的选择

  • 选择数组:可以直接使用数组下标进行哈希映射(如使用小写字母的相对位置映射),元素的分散程度不是很大
  • 选择set:元素较为分散,且映射方式相对复杂
  • 选择map:需要存储键值对

上述内容是我在第一次学习哈希表和哈希法时的笔记,欢迎大家指正和补充!!

  • 23
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值