hot 100 (1)—— 两数之和(哈希)

一、常规暴力解法(两层for循环)

java 复制代码
class Solution {
    public int[] twoSum(int[] nums, int target) {
        for(int i=0;i<nums.length;i++){
            for(int j=i+1;j<nums.length;j++){
                if(nums[i] + nums[j] == target){
                    return new int[]{i,j};
                }
            }
        }
        return new int[]{};
    }
}
  1. 代码功能解释

这段Java代码实现了两数之和问题的解法:给定一个整数数组nums和一个目标值target,找到数组中两个数相加等于target的索引,并返回这两个索引(题目默认有且只有一个有效答案)。

核心逻辑是通过两层循环遍历数组:

• 外层循环(i)从数组第一个元素开始,依次取每个元素作为"基准元素";

• 内层循环(j)从i+1开始,遍历"基准元素"之后的所有元素,检查两者之和是否等于target;

• 若满足条件,直接返回包含i和j的数组;若遍历结束未找到,返回空数组(实际题目场景下不会触发)。

  1. 时间复杂度

• 复杂度级别:O(n²)(n为数组nums的长度)。

• 原因:外层循环执行n次,内层循环针对每个i分别执行n-1、n-2...1次,整体执行次数约为n(n-1)/2,属于平方级别的时间开销,数据量增大时效率会显著下降。

  1. 空间复杂度

• 复杂度级别:O(1)(常数级)。

• 原因:算法仅使用了i、j两个循环变量,未额外创建与数组长度n相关的存储结构(如数组、哈希表等),占用的内存空间不随输入数据量变化。

二、最优解法(HashMap)

java 复制代码
class Solution {
    public int[] twoSum(int[] nums, int target) {
        Map<Integer,Integer> map = new HashMap<>();
        for(int i=0;i<nums.length;i++){
            if(map.containsKey(target - nums[i])){
                return new int[]{map.get(target - nums[i]),i};
            }
            map.put(nums[i],i);
        }
        return new int[]{};
    }
}
  1. 代码功能解释

这段Java代码是两数之和的优化解法,核心思路是用哈希表(HashMap)存储已遍历元素,实现"一次遍历找答案",步骤如下:

◦ 初始化HashMap,键存数组元素值,值存该元素的原始索引;

◦ 遍历数组(索引i从0开始):

  1. 计算"目标补数":target - nums[i](即与当前元素相加等于target的数);

  2. 检查哈希表中是否存在该"补数":若存在,直接返回[补数的索引, 当前索引i](补数先存入,索引更小);

  3. 若不存在,将当前元素nums[i]和其索引i存入哈希表,继续遍历;

◦ 题目默认有唯一解,遍历结束前必返回结果,末尾空数组仅为语法补充。

  1. 时间复杂度

• 复杂度级别:O(n)(n为数组nums的长度)。

• 原因:仅需遍历数组一次,哈希表的containsKey和put操作均为O(1) 平均时间复杂度,整体无嵌套循环,时间开销随数组长度线性增长。

  1. 空间复杂度

• 复杂度级别:O(n)(线性级)。

• 原因:最坏情况下(如答案在数组末尾),需将数组中n-1个元素存入哈希表,哈希表占用空间随数组长度线性增长,故为O(n)。

三、额外解法(二维数组dp + 双指针)

java 复制代码
class Solution {
    public int[] twoSum(int[] nums, int target) {
        int n = nums.length;
        int[][] dp = new int[n][2];
        for(int i=0;i<n;i++){
            dp[i][0] = nums[i];
            dp[i][1] = i;
        }
        Arrays.sort(dp, (a,b) -> a[0] - b[0]);//一定要排序!!!,并且要先填充dp再排序!!!
        int left = 0;
        int right = nums.length - 1;
        while(left < right){
            int sum = dp[left][0] + dp[right][0];
            if(sum == target){
                return new int[]{dp[left][1],dp[right][1]};
            }else if(sum < target){
                left++;
            }else{
                right--;
            }
        }
        return new int[]{};
    }
}
  1. 代码功能解释

这段Java代码同样解决两数之和问题,核心思路是"先排序,再双指针",步骤如下:

  1. 保存原始索引:创建二维数组dp(每行存[数组元素值, 元素原始索引]),避免排序后丢失原始索引(题目要求返回原始索引);

  2. 排序数组:按dp每行的第一个元素(即原数组的数值)升序排序,这是双指针查找的前提;

  3. 双指针查找:左指针left从数组开头出发,右指针right从数组末尾出发,计算两指针指向元素的和:

◦ 若和等于target,直接返回两元素的原始索引(dp[left][1]和dp[right][1]);

◦ 若和小于target,左指针右移(增大数值);

◦ 若和大于target,右指针左移(减小数值);

  1. 若遍历结束未找到(题目默认有解,此情况不触发),返回空数组。

  2. 时间复杂度

• 复杂度级别:O(n log n)(n为数组nums的长度)。

• 原因:算法耗时主要在排序步骤(Arrays.sort采用双轴快排,时间复杂度为O(n log n)),后续双指针遍历仅需O(n),整体由排序主导,故为O(n log n)。

  1. 空间复杂度

• 复杂度级别:O(n)(线性级)。

• 原因:额外创建了二维数组dp,其长度与原数组nums一致(n行2列),占用空间随n线性增长,故为O(n)。

相关推荐
又是忙碌的一天8 分钟前
Java IO流
java·开发语言
程序员buddha10 分钟前
springboot-mvc项目示例代码
java·spring boot·mvc
不懂英语的程序猿41 分钟前
【Java 工具类】Java通过 TCP/IP 调用斑马打印机(完整实现)
java
蒙奇D索大1 小时前
【算法】递归算法的深度实践:从布尔运算到二叉树剪枝的DFS之旅
笔记·学习·算法·leetcode·深度优先·剪枝
卡提西亚2 小时前
C++笔记-25-函数模板
c++·笔记·算法
ghie90902 小时前
MATLAB/Simulink水箱水位控制系统实现
开发语言·算法·matlab
多多*2 小时前
分布式系统中的CAP理论和BASE理论
java·数据结构·算法·log4j·maven
sg_knight2 小时前
Docker 实战:如何限制容器的内存使用大小
java·spring boot·spring·spring cloud·docker·容器·eureka
yuan199972 小时前
基于粒子群优化(PSO)算法的PID控制器参数整定
算法
合作小小程序员小小店3 小时前
web网页开发,在线考勤管理系统,基于Idea,html,css,vue,java,springboot,mysql
java·前端·vue.js·后端·intellij-idea·springboot