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)。

相关推荐
梵得儿SHI3 小时前
Java 注解与反射实战:自定义注解从入门到精通
java·开发语言·注解·自定义注解·元注解·控制注解的作用·声明式编程思想
_dindong4 小时前
牛客101:链表
数据结构·c++·笔记·学习·算法·链表
速易达网络4 小时前
Java Web登录系统实现(不使用开发工具)
java·开发语言·前端
wuk9984 小时前
基于位置式PID算法调节PWM占空比实现电机转速控制
单片机·嵌入式硬件·算法
不到满级不改名4 小时前
EM算法 & 隐马尔可夫模型
算法
悟能不能悟4 小时前
java重构旧代码有哪些注意的点
java·开发语言·重构
怪兽20145 小时前
Redis过期键的删除策略有哪些?
java·数据库·redis·缓存·面试
workflower8 小时前
单元测试-例子
java·开发语言·算法·django·个人开发·结对编程
YuanlongWang8 小时前
C# 基础——装箱和拆箱
java·开发语言·c#