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;
    }
}
相关推荐
罗超驿1 小时前
10.滑动窗口解决:无重复字符的最长子串 | LeetCode 3 Java 题解
java·算法·leetcode·面试
菜菜的顾清寒1 小时前
HOT100力扣(40) 动态规划-爬楼梯
算法·leetcode·动态规划
m沐沐2 小时前
【机器学习】聚类算法-K-means聚类
人工智能·python·算法·机器学习·pycharm·kmeans·聚类
z落落2 小时前
C# Dictionary 字典集合+数组、List、Dictionary 三者终极对比
算法
醇氧2 小时前
排队论(牛吃草问题)解题全解析
算法
蓝速科技2 小时前
3D 数字人全息舱算力部署方案对比:本地 X86 独显架构与云端 RK 架构怎么选才好
数据结构·人工智能·算法·架构·排序算法
qingyulee2 小时前
集成学习、聚类算法
算法·聚类·集成学习
lqqjuly2 小时前
机器人状态估计与 SLAM—概率推理到 simultaneous Localization and Mapping
算法·机器人
清 澜2 小时前
基于 LangChain 从零搭建知识库问答系统
人工智能·职场和发展·大模型·agent·知识库