leetcode 18. 四数之和(优质解法)

news/2024/7/6 3:32:15

代码:

class Solution {
    public List<List<Integer>> fourSum(int[] nums, int target) {
        List<List<Integer>> lists=new ArrayList<>();

        int length=nums.length;
        Arrays.sort(nums);

        for(int i=0;i<=length-4;){
            for(int j=i+1;j<=length-3;){
                //先固定两个数,再采用双指针和利用有序数组单调性来找到符合条件的另外两个数
                //加减操作很可能会溢出,所以最好用 long 类型
                long LRTarget=(long) target-nums[i]-nums[j];

                int left=j+1;
                int right=length-1;
                while (left<right){
                    long newLR=nums[left]+nums[right];
                    if(newLR<LRTarget){
                        left++;
                    }else if(newLR>LRTarget){
                        right--;
                    }else {
                        //符合条件,记录当前的 nums[i],nums[j],nums[left],nums[right]
                        lists.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--;
                        }
                    }
                }
                //当前固定的 num[j] 的所有情况都找到了
                j++;
                while (j<=length-3&&nums[j]==nums[j-1]){
                    j++;
                }
            }

            //当前固定的 num[i] 的所有情况都找到了
            i++;
            while (i<=length-4&&nums[i]==nums[i-1]){
                i++;
            }
        }

        return lists;
    }
}

题解:

        我们要在数组中选出相加为 0 的三个数,要选出符合条件的多个数,我们可以尝试采用先排序,利用有序数组的单调性和双指针的方式解决

        四数之和的求解方法和三数之和几乎一样,只是多了一个固定的数而已,推荐先看leetcode 15. 三数之和(优质解法)

          首先对于示例 ,-1,-1,3,0,5,-1,3,0, target = 1 经过排序以后得到 -1.-1,-1,0,0,3.,3,5由于此时我们要获取 4 个数,而采用双指针的方式只能探讨两个数的选择,所以我们可以先固定两个数,先用指针 i 指向要固定的数 -1,指针 j 指向要固定的数 -1,此时我们只需要在 j 右边的区间内找到两个数,使 nums[ L ] + nums[ R ]= target - nums[ i ] - nums[ j ] 即可,此时 nums[ L ] + nums[ R ]=1-(-1)-(-1)=3

        如下图,让指针 L 指向右边区间最小的数,指针 R 指向右边区间最大的数,此时 nums[ L ] + nums[ R ] = -1+5 = 4 > 3, 就需要取的两数之和小一点,此时 L 指针已经是区间中最小的了,所以需要淘汰大的数 5, R - -

-1        -1        -1        0        0        3        3        5

 i           j          L                                                R

        此时 nums[ L ] + nums[ R ] = -1+3=2 < 3 ,就需要取的两数之和大一点,此时 R 指针已经是区间中最大的了,所以需要淘汰小的数 -1,L++

-1        -1        -1        0        0        3        3        5

 i           j          L                                      R

         此时 nums[ L ] + nums[ R ] = 0+3=3 == 3,符合条件,就记录当前 nums[ i ] , nums[ j ] ,nums[ L ] ,nums[ R ] 的值,由于需要找出所有的情况,所以现在还不能停止,让 L++,R- -,继续寻找符合条件的数据

-1        -1        -1        0        0        3        3        5

 i           j                    L                            R

        题目中有要求,要去除重复的数据,当我们排序以后相同的数据都会靠在一起,我们上一轮已经将 nums[ L ] 为 0 的值获取了,此时 nums[ L ] 依然为 0,即使有符合条件的 nums[ R ] 也是重复的要排除掉,所以当 nums[ L ] == nums[ L-1 ] 时,就直接 L ++ 跳过(在跳过时要注意边界问题),对于 R 指针也是一样的处理,当 nums[ R ] == nums[ R+1 ] 时,就直接 R- - 跳过

-1        -1        -1        0        0        3        3        5

 i           j                              L        R        

        当 L == R 时就说明当前固定的 nums[ j ] 的情况已经全部发现了,就需要 j++,但其中也涉及到数据重复的问题,j ++ 以后指向的值依然为 -1 ,代表要在右边的区间寻找两个和为 3 的数,但之前已经寻找过了,现在再找一遍也只会找到同样的数据,所以当 nums[ j ] == nums[ j+1 ] 时就直接 j++ 跳过,对于下标 i 也一样,当nums[ i ] == nums[ i+1 ] 时就直接 i++ 跳过

-1        -1        -1        0        0        3        3        5

 i                      j                            R        

                                                      L         


http://lihuaxi.xjx100.cn/news/1854146.html

相关文章

1657.确定两个字符串是否接近

​​题目来源&#xff1a; leetcode题目&#xff0c;网址&#xff1a;1657. 确定两个字符串是否接近 - 力扣&#xff08;LeetCode&#xff09; 解题思路&#xff1a; 当一个字符串中出现的字符全部在另一个字符串中出现并且 两字符串各字符出现次数排序后的 有序序列相同 时&a…

NX二次开发UF_MTX2_vec_multiply 函数介绍

文章作者&#xff1a;里海 来源网站&#xff1a;https://blog.csdn.net/WangPaiFeiXingYuan UF_MTX2_vec_multiply Defined in: uf_mtx.h void UF_MTX2_vec_multiply(const double vec [ 2 ] , const double mtx [ 4 ] , double vec_product [ 2 ] ) overview 概述 Returns…

Spring Security OAuth2之认证服务、资源服务、web安全配置服务加载优先级详解

order的值越小&#xff0c;类的优先级越高&#xff0c;IOC容器就会优先加载&#xff0c;上面的优先级是&#xff1a;认证服务器配置&#xff08;0&#xff09;>资源服务器配置&#xff08;3&#xff09;>web安全服务配置&#xff08;100&#xff09;在做资源权限配置的时…

基于AT89C51单片机的电子闹钟设计

1&#xff0e;设计任务 利用AT89C51单片机为核心控制元件,设计一个电子闹钟&#xff0c;设计的系统实用性强、操作简单&#xff0c;实现了智能化、数字化。 &#xff08;1&#xff09;按开始键自动进入时间显示&#xff0c;开始为0&#xff0c;按K1键进入更改时间&#xff0c…

深信服技术认证“SCSA-S”划重点:SQL注入漏洞

为帮助大家更加系统化地学习网络安全知识&#xff0c;以及更高效地通过深信服安全服务认证工程师考核&#xff0c;深信服特别推出“SCSA-S认证备考秘笈”共十期内容&#xff0c;“考试重点”内容框架&#xff0c;帮助大家快速get重点知识~ 划重点来啦 深信服安全服务认证工程师…

SNAT、DNAT

一.NAT NAT: network address translation&#xff0c;支持PREROUTING&#xff0c;INPUT&#xff0c;OUTPUT&#xff0c;POSTROUTING四个链 请求报文&#xff1a;修改源/目标IP&#xff0c; 响应报文&#xff1a;修改源/目标IP&#xff0c;根据跟踪机制自动实现 NAT的实现分…

深入学习redis-基于Jedis通过客户端操作Redis

目录 redis客户端&#xff08;JAVA&#xff09; 配置 引入依赖 建立连接 常用命令实现 get/set exists/del keys expire和ttl type 字符串&#xff08;String&#xff09; mget和mset getrange和setrange append incr和decr 列表&#xff08;list&#xff09; …

如何查看电脑内存?Windows 和 Mac 方法不同

Windows 系统查看内存方法 在 Windows 操作系统中我们查看电脑内存在哪里查呢&#xff1f;下面总结的 3 种查看电脑内存的方法都可以使用&#xff1a;使用任务管理器&#xff1a;任务管理器是 Windows 中一个强大的工具&#xff0c;可用于监视和管理计算机的性能和资源使用。使…