当前位置: 首页 > news >正文

双指针-三数之和

三数之和

给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 
满足 i != j、i != k 且 j != k ,同时还满足 nums[i] + nums[j] + nums[k]
 == 0 。请你返回所有和为 0 且不重复的三元组。

注意:答案中不可以包含重复的三元组。

输入:整型数组
输出:二元列表
思路:先对数组进行排序,然后循环排序后的数组,再定义双指针,进行移动,关键是对于去重的操作,

class Solution {
    public List<List<Integer>> threeSum(int[] nums) {
        //定义返回值的二元列表
        List<List<Integer>> result = new ArrayList<>();
        //先对数组nums进行排序
        Arrays.sort(nums);
        //当第一个数值大于0,则直接返回
        if(nums[0] > 0){
            return result;
        }
        //然后进行for循环遍历每一个数组元素
        for(int i = 0;i < nums.length;i++){
            //定义双指针
            int l = i + 1;
            int r = nums.length - 1;
            if (i > 0 && nums[i] == nums[i - 1]) {  // 去重a
                continue;
            }
            //应该在循环中不断变换双指针
            while(l < r){
                //判断值与零的大小关系
                if(nums[i] + nums[l] + nums[r] > 0){
                //说明右边大了
                    r--;
                }else if(nums[i] + nums[l] + nums[r] < 0){
                    //说明左边小了
                    l++;
                }else{
                    //说明等于0
                    result.add(new ArrayList<>(Arrays.asList(nums[i],nums[l],nums[r])));
                    while (r > l && nums[r] == nums[r - 1]) r--;
                    while (r > l && nums[l] == nums[l + 1]) l++;
                    r--; 
                    l++;
                }
            }
        }
        return result;
    }
}

去重逻辑的思考
其实就是左中右三个数的去重,i,l,r索引所对应的值

  • 对于i的去重应该使用nums[i] == nums[i-1]判断,如果写成
if (nums[i] == nums[i + 1]) { // 去重操作
    continue;
}

那我们就把 三元组中出现重复元素的情况直接pass掉了。 例如{-1, -1 ,2} 这组数据,当遍历到第一个-1 的时候,判断 下一个也是-1,那这组数据就pass了。
那么应该这么写:

if (i > 0 && nums[i] == nums[i - 1]) {
    continue;
}

这么写就是当前使用 nums[i],我们判断前一位是不是一样的元素,在看 {-1, -1 ,2} 这组数据,当遍历到 第一个 -1 的时候,只要前一位没有-1,那么 {-1, -1 ,2} 这组数据一样可以收录到 结果集里。

  • 只有在三数之和等于零是才有可能出现重复的情况,所以以下逻辑不合理,并无卵用
while (right > left) {
    if (nums[i] + nums[left] + nums[right] > 0) {
        right--;
        // 去重 right
        while (left < right && nums[right] == nums[right + 1]) right--;
    } else if (nums[i] + nums[left] + nums[right] < 0) {
        left++;
        // 去重 left
        while (left < right && nums[left] == nums[left - 1]) left++;
    } else {
    }
}

相关文章:

  • 【YOLOv11改进- 主干网络】YOLOv11+CSWinTransformer: 交叉窗口注意力Transformer助力YOLOv11有效涨点;
  • MongoDB:记一次数据迁移经验
  • JavaSE的基础语法(5)
  • PostgreSQL如何关闭自动commit
  • 基于Python的Flask微博话题舆情分析可视化系统
  • SaaS 平台开发要点
  • javascript-es6 (四)
  • 【NLP251】命名实体实战(基于Transformer分类)
  • 【BUUCTF逆向题】[ACTF新生赛2020]Splendid_MineCraft(SMC代码混淆)
  • 【强化学习】强化学习(Reinforcement Learning, RL)详解
  • SpringBoot+uniApp日历备忘录小程序系统 附带详细运行指导视频
  • 459重复的子字符串(substr)
  • Word中的文档信息域
  • Java语言介绍
  • cap2:1000分类的ResNet的TensorRT部署指南(python版)
  • Linux:深入了解进程信号(上)
  • (7/100)每日小游戏平台系列
  • 力扣做题记录 (二叉树)
  • MySQL 插入替换语句(replace into statement)
  • 配置Open-R1,评测第三方蒸馏模型的性能1
  • 生命与大海相连:他在300多米的深海行走,在沉船一线打捞救援
  • 体坛联播|欧冠巴萨3比3战平国米,柯洁未进入国家集训队
  • 美国务院宣布新一轮与伊朗相关的制裁
  • 国务院食安办:加强五一假期食品生产、销售、餐饮服务环节监管
  • 医学统计专家童新元逝世,终年61岁
  • 国家能源局通报上月投诉情况:赤峰有群众反映电费异常增高,已退费