首页 > 编程语言 >算法之哈希表

算法之哈希表

时间:2024-12-23 10:20:34浏览次数:3  
标签:map key nums int value ++ 算法 哈希

有效的字母异位词

思想和滑动窗口找子串很像都是将字符串中的字符统计情况放到数组中,这个不需要进行窗口的移动,因为两个的长度必须相同。

快乐数

没有注意到提示无限循环,即当平方和重复出现时,不可能为快乐数,此时使用set对平方和进行存储,只有和前面都不重复时才会将此时的平方和放到set中,若重复则不为快乐数,当平方和为1时则为快乐数。

查看代码
 public boolean isHappy(int n) {
        Set<Integer> record=new HashSet<>();
        while (n!=1 && !record.contains(n)){
            record.add(n);
            int sum=0;
            while (n!=0){
                int temp=n%10;
                sum+=temp*temp;
                n=n/10;
            }
            n=sum;
        }
        if (n==1){
            return true;
        }else {
            return false;
        }
    }

两数之和(使用减法)

暴力求解要两层循环,使用map可以只遍历一次,哈希表可以用来快速判断所找的值存不存在

本题中,使用map的key部分用来存放数值,value部分用来存放下标。

每指向数组中的一个元素,就去map中查找对应的target减去元素值,如果有则找到,如果没有将该数组元素的值存到map中。

其中java操作:map.get("key")得到key对应的value值

       map.put("key","value")向map中添加键值对

         map.contain("key")查找该key对应的键值对在不在map中

查看代码
 public int[] twoSum(int[] nums, int target) {
        HashMap<Integer, Integer> numIndex = new HashMap<>();
        int[] result=new int[2];
        for (int i = 0; i < nums.length; i++) {
            int number=target-nums[i];
            if (numIndex.containsKey(number)){
                result[0]=numIndex.get(number);
                result[1]=i;
            }else {
                   numIndex.put(nums[i],i);
            }

        }
        return result;
    }

 

四数相加

和“两数之和”的区别在于将两个数组的和存放在map里面的key中,和出现的次数存放在map里面的key中。

查看代码

public int fourSumCount(int[] nums1, int[] nums2, int[] nums3, int[] nums4) {
        HashMap<Integer, Integer> map = new HashMap<>();
        for (int i = 0; i < nums1.length; i++) {
            for (int j = 0; j < nums2.length; j++) {
                int sum12=nums1[i]+nums2[j];
                if (map.containsKey(sum12)){
                    Integer value = map.get(sum12);
                    map.put(sum12,value+1);//不能写成value++
                }else {
                    map.put(sum12,1);
                }
            }

        }

        int count =0;
        for (int i = 0; i < nums3.length; i++) {
            for (int j = 0; j < nums4.length; j++) {
                int sum34=nums3[i]+nums4[j];
                if (map.containsKey(-sum34)){
                    count=count+map.get(-sum34);
                }
            }
        }
        return count;
    }

出现问题:不能写成value++,这样会先用原来的值把map更新,然后再将value+1;

 

三数之和

此题最好使用快慢指针,使用哈希表很难判断里面的元素到底重不重。

原因:如果按照四数相加的思想先将数组中的元素所有都不重复的两两相加,将结果保存在map中作为key,对应两个数的下标作为value,第三个数选择不同于这两个下标的数,然后将结果和为0的下标放入结果中,这么做判重会非常复杂,没法保证List<List>中元素List里的数不相同,只能保证下标都不相同。

而二数之和和四数之和一个是返回下标,另一个是满足条件的次数,不需要满足数值不重复,则只用哈希表即可。

 

其中Java基础:for中每次循环最后都会i++,即便contine了,i++也会执行

       数组升序排序:Arrays.sort(nums)

查看代码
 public List<List<Integer>> threeSum(int[] nums) {
        List<List<Integer>> result=new ArrayList<>();
        Arrays.sort(nums);//对数组进行升序排序

        int left;
        int right;

        for (int i=0;i<nums.length;i++){

            left=i+1;
            right=nums.length-1;

            if (i>0 && nums[i]==nums[i-1]){
                continue;
            }
            while (left<right){
                if (nums[i]+nums[left]+nums[right]>0){
                    right-=1;
                } else if (nums[i]+nums[left]+nums[right]<0) {
                    left+=1;
                }else {

                    result.add(Arrays.asList(nums[i],nums[left],nums[right]));

                    while (nums[left]==nums[left+1] && left+1<nums.length-1){
                        left+=1;
                    }
                    left++;
                    right--;
                }
            }
        }

        return result;
    
    }

 

四数相加

思路和三数相加相同,因为要求也是输出不重复的结果

遇到的问题:

  1. 当指针所指向的四个数之和满足target之时,进行完while重复判断之后,需要再单独的right--,left++。因为此时只是找到了不重复的前一个值而不是当前值,需要再进行操作使两个指针达到不重复值。
  2. 在重复判断时,必须要将left<right的判断条件放在前面,这样如果前者不满足情况将不会再判断后者。
  3. 当数值相加以及超过int型时,此时判断可能出错,比如当数组为1000000000,1000000000,1000000000,1000000000,而target为-294967296,此时判断相等,因此需要在判断中强制转换int类型为long,java中long占64位(8个字节),int占32位(4个字节)
  4. 关于不重复的判断,必须要是nums[i]==nums[i-1]而不是能是nums[i]==nums[i+1]。

 

查看代码
  public List<List<Integer>> fourSum(int[] nums, int target) {
        Arrays.sort(nums);
        List<List<Integer>> result=new ArrayList<>();

        int left;
        int right;
        for (int i = 0; i < nums.length; i++) {
            if (i>0 &&nums[i]==nums[i-1]){
                continue;
            }
            for (int j = i+1; j <nums.length ; j++) {
                if (j-1>i && nums[j]==nums[j-1]){
                    continue;
                }
                 left=j+1;
                 right=nums.length-1;
                 while (left<right){
                     if ((long)nums[left]+nums[right]+nums[i]+nums[j]>(double) target){
                         right--;
                     }else if ((double)nums[left]+nums[right]+nums[i]+nums[j]<(double)target){
                         left++;
                     }else {
                         result.add(Arrays.asList(nums[i],nums[j],nums[left],nums[right]));
                         while (left<right&& nums[left]==nums[left+1]) left++;//left<roght必须放在前面
                         while (left<right&&nums[right]==nums[right-1]) right--;
                         left++;//因为上面只是用来找到不重复的前一个值,而不是不重复的当前值 所以最后还是要再left++right--
                         right--;
                     }
                 }
            }
        }
        return result;
    }

 

标签:map,key,nums,int,value,++,算法,哈希
From: https://www.cnblogs.com/exo123/p/18609675

相关文章

  • 深入探索人工智能的技术热点:生成式AI、强化学习与AI算法优化
    人工智能(AI)技术在不断发展中,带来了许多突破性的进展。我们看到了生成式AI在图像、文本生成等领域的广泛应用,也见证了强化学习在复杂决策问题中的成功实践。同时,随着AI技术逐渐走向实际应用,算法优化与效率提升成了新的技术焦点。在这篇博客中,我们将重点讨论目前在人工智能领域的......
  • 算法——排序算法(冒泡、选择、归并、堆排序)
    排序算法——冒泡排序(BubbleSort)排序算法——选择排序(SelectionSort)排序算法——插入排序(InsertionSort)排序算法——堆排序(HeapSort)排序算法——归并排序(MergeSort)......
  • 哈希
    哈希概述哈希最优秀的点,或者是用哈希的目的就是比较两个东西做到O(1)的复杂度这个东西可能是:字符串,数列,甚至是树,图,你能想到的东西都可以根据不同哈希函数映射成数字,只不过难度有所不同此篇文章特别介绍用的比较多的哈希方法:字符串哈希,数列哈希,树哈希也引出......
  • 永磁同步电机无速度算法--全阶滑模观测器
    一、原理介绍在采用传统滑模观测器求取电机角度时通常存在系统抖振、低通滤波器导致角度相位滞后、角度的求取等问题。针对上述问题,本文采用全阶滑模观测器,该全阶滑模观测器具有二阶低通滤波器的特性,能有效滤除反电动势中的高频噪声,无需使用低通滤波器;在计算估计的电流时传统......
  • 【论文投稿】探秘计算机视觉算法:开启智能视觉新时代
     【即将截稿!快检索】第三届教育科学与社会文化国际学术会议(ESSC2024)_艾思科蓝_学术一站式服务平台更多学术会议请看:https://ais.cn/u/nuyAF3 目录引言一、计算机视觉算法基石:图像基础与预处理二、特征提取:视觉信息的精华萃取三、目标检测:从图像中精准定位目标四、......
  • 在SpringBoot项目中接入sensitive-word实现敏感词过滤(DFA算法、为敏感词打上标签、忽
    文章目录1.前言2.敏感词过滤的常见解决方案3.DFA算法3.1什么是DFA算法3.2DFA算法的原理3.2.1数据是如何存储的3.2.2数据是如何检索的3.3DFA算法的应用场景4.sensitive-word简介4.1什么是sensitive-word4.2sensitive-word的官网4.3sensitive-word的性能5.......
  • (Matlab实现)K-means算法及最佳聚类数目的确定
    目录摘要:1.K-means算法2.Calinski-HarabaszCriterion(卡林斯基-哈拉巴斯指标,CH值)3.Davies-BouldinCriterion(戴维斯-博尔丁指标,DB值)4.GapValue(Gap值)5.SilhouetteCoefficient(轮廓系数)6.基于Matlab的K-means聚类及最佳聚类数选取结果:各种指标评价图像:K-means聚类结果......
  • YOLOv11/10/8算法改进【NO.158】使用一种名为 PRepBN 的新方法,在训练过程中逐步用重新
      前  言    YOLO算法改进系列出到这,很多朋友问改进如何选择是最佳的,下面我就根据个人多年的写作发文章以及指导发文章的经验来看,按照优先顺序进行排序讲解YOLO算法改进方法的顺序选择。具体有需求的同学可以私信我沟通:首推,是将两种最新推出算法的模块进行融合形......
  • 【初阶数据结构与算法】八大排序算法之交换排序(冒泡排序,快速排序---hoare、挖坑法、lo
    文章目录一、冒泡排序二、快速排序简介及其大致框架三、快排hoare版本子函数四、快排挖坑法子函数五、快排lomuto双指针子函数六、冒泡排序与快排的性能分析与比较一、冒泡排序   冒泡排序的命名是因为它的排序操作就像水平面在冒泡一样,当我们讲完冒泡排序就知道......
  • 斐波那契查找算法
    1,什么是斐波那契查找算法?    斐波那契查找算法(FibonacciSearch)是一种基于斐波那契数列的搜索算法。与二分查找算法相比,斐波那契查找更适用于没有直接索引访问的数据结构(如链表)。它通过使用斐波那契数列来逐步缩小搜索范围,从而找到目标元素的位置。斐波那契数列斐......