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

leetcode 18. 四数之和(优质解法)_第1张图片

代码:

class Solution {
    public List> fourSum(int[] nums, int target) {
        List> 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 (leftLRTarget){
                        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

题解:

        我们要在数组中选出相加为 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         

你可能感兴趣的:(leetcode,leetcode,算法,职场和发展)