8.【LeetCode 18】四数之和 —— Java 排序 + 双指针解法详解

目录

[1. 题目解析](#1. 题目解析)

[18. 四数之和​](#18. 四数之和)

[示例 1:](#示例 1:)

[示例 2:](#示例 2:)

[2. 算法原理](#2. 算法原理)

[解法一:排序 + 暴力枚举 + 利用 set 去重](#解法一:排序 + 暴力枚举 + 利用 set 去重)

[解法二:排序 + 双指针](#解法二:排序 + 双指针)

处理细节问题:

[3. 编写代码](#3. 编写代码)


OJ链接:https://leetcode.cn/problems/4sum/description/


1. 题目解析

18. 四数之和

难度:中等

给你一个由 n 个整数组成的数组 nums,和一个目标值 target。请你找出并返回满足下述全部条件且不重复的四元组 [nums[a], nums[b], nums[c], nums[d]](若两个四元组元素一一对应,则认为两个四元组重复):

  • 0 <= a, b, c, d < n

  • abcd互不相同

  • nums[a] + nums[b] + nums[c] + nums[d] == target

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


示例 1:

输入:nums = [1,0,-1,0,-2,2], target = 0

输出:[[-2,-1,1,2],[-2,0,0,2],[-1,0,0,1]]


示例 2:

输入:nums = [2,2,2,2,2], target = 8

输出:[[2,2,2,2]]


2. 算法原理

解法一:排序 + 暴力枚举 + 利用 set 去重


解法二:排序 + 双指针

  • 依次固定一个数 a
  • a后面的区间内,利用"三数之和"找到三个数,使这三个数的和等于 target - a即可。

进一步细化:

复制代码
1. 依次固定一个数 b;
2. 在 b 后面的区间内,利用"双指针"找到两个数,
   使这两个数的和等于 target - a - b 即可。

处理细节问题:

  1. 不重

  2. 不漏


3. 编写代码

java 复制代码
class Solution {
    public List<List<Integer>> fourSum(int[] nums, int target) {
        List<List<Integer>> ret = new ArrayList<>();

        //排序
        Arrays.sort(nums);

        //四数之和
        int n = nums.length;
        for(int i = 0; i < n - 3; ){//固定i
            //三数之和
            for(int j = i + 1; j < n - 2; ){//固定j
                //两數之和
                int left = j + 1, right = n - 1;
                //避免溢出,这里需要使用Long!!!
                long aim = (long)target - nums[i] - nums[j];
                while(left < right){
                    int sum = nums[left] + nums[right];
                    if(sum > aim) right--;
                    else if(sum < aim) left++;
                    else{
                        ret.add(Arrays.asList(nums[i],nums[j],nums[left],nums[right]));
                        //不漏
                        left++; right--;
                        //去重一
                        while(left < right && nums[left] == nums[left - 1]) left++;
                        while(left < right && nums[right] == nums[right + 1]) right--;
                    }
                }
                //去重二
                j++;
                while(j < n && nums[j] == nums[j - 1]) j++;
            }
            //去重三
            i++;
            while(i < n && nums[i] == nums[i - 1]) i++;
        }
        return ret;
    }
}
相关推荐
Jack203 小时前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树5 小时前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
JieE21221 小时前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE21221 小时前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
vivo互联网技术1 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦1 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
用户497863050731 天前
(一)小红的数组操作
算法·编程语言
怕浪猫1 天前
Electron 系列文章封面图
算法·架构·前端框架
徐小夕1 天前
JitWord 3.0 正式发布,高精度Word异构解析+复杂组件兼容,打造web端协同Word编辑器
前端·vue.js·算法