三数之和(Java实现)

news2025/1/11 23:48:14

文章目录

  • 思路
  • 1.暴力算法(超出时间限制)
    • ==解题思路==
    • ==复杂度==
  • 2.双指针算法
    • ==解题思路:==
    • ==注意点==
    • ==复杂度==

Problem: 15. 三数之和

在这里插入图片描述

思路

1.暴力算法
2.双指针算法

1.暴力算法(超出时间限制)


class Solution {
    public List<List<Integer>> threeSum(int[] nums) {
        // 检查输入数组是否为空或长度小于等于2,如果是,则返回空列表
        if (nums == null || nums.length <= 2) {
            return Collections.emptyList();
        }
        // 对输入数组进行排序
        Arrays.sort(nums);
        // 初始化结果列表
        List<List<Integer>> ret = new ArrayList<>();
        // 第一层循环遍历数组中的每个元素
        for (int i = 0; i < nums.length; i++) {
            // 第二层循环遍历位于当前元素之后的元素
            for (int j = i + 1; j < nums.length; j++) {
                // 第三层循环遍历位于第二个元素之后的元素
                for (int k = j + 1; k < nums.length; k++) {
                    // 判断当前三元组是否满足和为0的条件
                    if (nums[i] + nums[j] + nums[k] == 0) {
                        // 创建一个新的列表并将满足条件的三元组添加到其中
                        List<Integer> value = new ArrayList<>();
                        value.add(nums[i]);
                        value.add(nums[j]);
                        value.add(nums[k]);
                        // 判断结果列表是否已经包含该三元组,如果不包含,则将其添加到结果列表中
                        if (!ret.contains(value)) {
                            ret.add(value);
                        }
                    }
                }
            }
        }
        // 返回包含满足条件的三元组的结果列表
        return ret;
    }
}

解题思路

暴力解法通过三层循环遍历数组中的所有三元组,并检查它们是否满足和为 0 的条件。当找到一个满足条件的三元组时,首先创建一个新的列表并将三个元素添加到其中。然后检查结果列表是否已经包含这个三元组,如果没有,则将其添加到结果列表中。最后,返回包含所有满足条件的不重复三元组的结果列表。

复杂度

  • 时间复杂度:

暴力算法的时间复杂度主要来自于三层嵌套循环。由于每一层循环都需要遍历数组中的元素,所以暴力算法的时间复杂度为 O(n^3),其中 n 是输入数组的长度。

  • 空间复杂度:

除了存储结果列表之外,暴力算法的空间复杂度较低。主要使用了几个变量(如 i、j 和 k)来控制循环。因此,算法的额外空间复杂度(除去结果列表)接近 O(1)。

2.双指针算法

class Solution {
    public static List<List<Integer>> threeSum(int[] nums) {
        // 对输入数组进行排序
        Arrays.sort(nums);

        // 创建一个 ArrayList 用于存储满足条件的三元组
        List<List<Integer>> result = new ArrayList<>();

        // 遍历数组,略过最后两个元素,因为我们需要至少三个元素来形成一个三元组
        for (int i = 0; i < nums.length - 2; i++) {
            // 如果当前元素大于0,不可能再找到和为0的三元组
            if (nums[i] > 0) {
                break;
            }

            // 跳过重复元素,以避免重复的三元组
            if (i > 0 && nums[i] == nums[i - 1]) {
                continue;
            }

            // 初始化双指针,一个指针从当前元素的下一个开始,另一个指针从数组的最后一个元素开始
            int left = i + 1;
            int right = nums.length - 1;

            // 当左指针小于右指针时,继续寻找满足条件的三元组
            while (left < right) {
                int sum = nums[i] + nums[left] + nums[right];

                // 根据三数之和与0的关系,移动左指针或右指针
                if (sum == 0) {
                    // 找到一个满足条件的三元组,添加到结果列表中
                    result.add(Arrays.asList(nums[i], nums[left], nums[right]));

                    // 跳过重复元素,以避免重复的三元组
                    while (left < right && nums[left] == nums[left + 1]) {
                        left++;
                    }
                    while (left < right && nums[right] == nums[right - 1]) {
                        right--;
                    }

                    // 找到一个满足条件的三元组后,同时移动左右指针继续寻找下一个三元组
                    left++;
                    right--;
                } else if (sum < 0) {
                    // 如果和小于0,需要增加和的值,因此将左指针向右移动
                    left++;
                } else {
                    // 如果和大于0,需要减小和的值,因此将右指针向左移动
                    right--;
                }
            }
        }

        // 返回包含满足条件的三元组的结果列表
        return result;
    }

}

解题思路:

使用双指针方法优化搜索过程。首先,对数组进行排序,这将使得重复元素相邻并允许我们在搜索过程中轻松地跳过它们。接下来,遍历数组并使用双指针技巧查找和为 0 的三元组。对于当前元素 nums[i],使用一个指针 left 从 i+1 开始,另一个指针 right 从数组末尾开始。根据三数之和与 0 的关系来移动指针:

  1. 如果三数之和等于 0,找到一个满足条件的三元组。将其添加到结果列表中,然后跳过左右指针所指向的重复元素,最后同时移动左右指针。
  2. 如果三数之和小于 0,意味着需要增加和的值。因此,将左指针向右移动以获得更大的值。
  3. 如果三数之和大于 0,意味着需要减小和的值。因此,将右指针向左移动以获得较小的值。

在遍历过程中,我们可以使用一些优化方法,例如在当前元素大于 0 时提前终止循环,因为排序后的数组中,大于 0 的元素无法与其他元素形成和为 0 的三元组。此外,当遇到连续相同的元素时,可以直接跳过以避免添加重复的三元组。

通过这种方法,我们可以有效地找到数组中所有满足条件的不重复三元组。相比暴力解法,双指针方法大大降低了时间复杂度。

注意点

在这个解法中,遍历数组时,我们略过最后两个元素,这是因为我们需要至少三个元素来形成一个三元组。当我们将索引 i 定位在倒数第三个元素上时,双指针 left 和 right 将分别指向倒数第二个和最后一个元素。在这种情况下,我们仍然可以形成一个三元组(nums[i]、nums[left] 和 nums[right])。

如果我们将索引 i 定位在倒数第二个元素或最后一个元素上,那么双指针 left 和 right 将无法指向有效的数组元素,从而无法形成三元组。在这种情况下,继续遍历是没有意义的。

为了避免这种情况并确保我们能够在遍历过程中始终找到有效的三元组,我们在循环中设置了条件 i < nums.length - 2,这样在循环中,索引 i 最多只会到达数组的倒数第三个元素。

复杂度

  • 时间复杂度:O(n^2)

双指针算法首先对数组进行排序,排序的时间复杂度为 O(nlogn)。接下来,它遍历数组的每个元素,并对每个元素使用双指针方法寻找满足条件的三元组。这一部分的时间复杂度为 O(n2)。因此,总的时间复杂度为 O(nlogn + n2)。在实际应用中,n2 项通常比 n*logn 项更显著,所以我们通常表示双指针算法的时间复杂度为 O(n2)。

  • 空间复杂度:O(m)

双指针算法的空间复杂度主要取决于存储结果的列表。在实际应用中,空间复杂度会受到满足和为 0 的三元组数量的影响。假设有 m 个满足条件的三元组,那么空间复杂度为 O(m)。除此之外,双指针算法使用了几个变量(如 i、left 和 right)来控制循环,因此算法的额外空间复杂度(除去结果列表)接近 O(1)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/482331.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

使用TrieTree(字典树)来实现敏感词过滤

使用TrieTree&#xff08;字典树&#xff09;来实现敏感词过滤 1. 字典树定义 字典树&#xff08;TrieTree&#xff09;&#xff0c;是一种树形结构&#xff0c;典型应用是用于统计&#xff0c;排序和保存大量的字符串&#xff08;但不仅限于字符串,如01字典树&#xff09;。…

Hive本地开发/学习环境配置

前提 hive依赖hadoop的相关组件&#xff0c;需要启动Hadoop的相关组件。 Hive 版本&#xff1a;3.1.3 Hadoop版本&#xff1a;3.3.4 hive-env.sh export HADOOP_HOME$HADOOP_HOME export HIVE_CONF_DIR/usr/local/Cellar/hive/3.1.3/libexec/conf export HIVE_AUX_JARS_PATH/…

micro-app的简单学习

本文承接上一篇手把手教你使用vue2搭建micro-app&#xff0c;对micro-app进行简单的认识与学习。 简述 因为上一篇只是对micro-app的搭建&#xff0c;并没有对具体的内容进行深入了解&#xff0c;所以本文是在上一篇文章代码的基础上对micro-app官网&#xff0c;的初步了解。…

Vue(标签属性:ref、配置项:props、混入mixin、插件、样式属性:scroped)

一、ref&#xff08;打标识&#xff09; 前面提及到了标签属性&#xff1a;keys 这里将了解ref&#xff1a;打标识 正常布置脚手架并创建入口文件main.js,引入组件 1. 可以给元素注册引用信息&#xff08;获取真实DOM&#xff09; 给一个按钮获取上方的dom的方法&#xff0c;方…

log4j2实现日志输出

引言 日志是我们在软件开发过程中非常重要的一个组成部分&#xff0c;它能够记录系统运行时的各种信息和异常&#xff0c;方便我们在需要的时候进行排查和调试。而Log4j2是目前最为流行的Java日志框架之一&#xff0c;它提供了丰富的日志输出方式和配置选项&#xff0c;可以满…

设计模式——装饰器模式(Decorator Pattern)

很久没有写博客了&#xff0c;最近也有很多事情要处理&#xff0c;也在努力的备考软件考试&#xff0c;正好模拟题中有一道关于装饰器模式的题&#xff0c;觉得还不错&#xff0c;所以特地写一篇文章希望能分享给小伙伴们。 装饰器模式的作用&#xff1a;允许向一个现有的对象…

vue3导入elcel表格并展示(使用xlsx插件+vite+element-plus)/js上传表格(js+xlsx)

表格内容(本博客演示的表格,这里其实可以更换任意表格,动态展示的) 安装插件xlsx npm install xlsx组件的所有代码(附解释) <script setup> import { ref } from "vue"; import * as XLSX from "xlsx"; // 把文件按照二进制进行读取 function read…

解密PyTorch动态计算图:打破深度学习束缚的秘密武器

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…

【C++】STL之priority_queue类源码剖析

目录 概述 算法 源码 PriorityQueue.h test.cpp 测试结果 概述 priority_queue&#xff1a;优先级队列&#xff0c;包含在头文件<queue>中 优先级队列类似于堆结构&#xff0c;优先级最高的元素被置为堆顶&#xff0c;最优先被弹出top()和删除pop() 优先级队列的…

Python凸包

文章目录 ConvexHullQG三维情况ConvexHull属性 ConvexHull ConvexHull是spatial中的一个类&#xff0c;主要功能是找到一组点的边缘&#xff0c;并做一个凸包。其必要的初始化参数为一个点集&#xff0c;点集格式为 n m n\times m nm维度的数组&#xff0c;n为点集中点的个数…

定位的特殊应用

注意&#xff1a;发生固定定位&#xff0c;绝对定位后&#xff0c;元素都变成了定位元素&#xff0c;默认高宽被内容撑开&#xff0c;则可以设置宽高&#xff1b;以下只针对绝对定位和固定定位的元素&#xff0c;不包括相对定位元素。 1.定位元素块的宽充满包含块 前提&#x…

封装建立-SMD封装

1. 看规格书&#xff0c;建立需要的焊盘&#xff0c;命名。注意padstack editor保存路径中不能有中文。 2.新建.dra工程&#xff0c;layout/pin 在里面筛选需要的焊盘。 3. 放置焊盘&#xff0c;需要计算精确坐标&#xff0c;allegro里command用x 0 0命令可以定位到原点。 4…

Python综合案例-学生数据可视化

近年来,数据分析和可视化已经成为了许多领域中的重要工具。在教育领域中,通过对学生的表现和行为进行数据分析和可视化,可以更好地了解学生的学习状态,发现问题、改进教学,并提高学生成绩。本文将介绍一个 Python 综合案例,使用 Pandas 和 Seaborn 库,对学生的数据进行清…

MySQL几种备份方式对比,你用对了吗?

各备份方法对比 备份数据的策略需要根据几种维度考虑 备份能承受最大丢失数据量 备份期间系统可以处于哪种情况&#xff08;不可用&#xff0c;部分可用&#xff0c;完全可用&#xff09; 数据恢复时长 需要恢复全量数据还是增量数据 备份数据的方法 逻辑备份&#xff1a;…

推荐算法实战项目:Deep Crossing 模型原理以及案例实战(附完整 Python 代码)

本文要介绍的Deep Crossing模型是由微软研究院在论文《Deep Crossing: Web-Scale Modeling without Manually Crafted Combinatorial Features》中提出的&#xff0c;它主要是用来解决大规模特征自动组合问题&#xff0c;从而减轻或者避免手工进行特征组合的开销。 Deep Cross…

推荐算法实战项目:DCN 原理以及案例实战(附完整 Python 代码)

本文要介绍的是由斯坦福大学联合Google的研究人员发表的论文《Deep & Cross Network for Ad Click Predictions》中提出的Deep&Cross模型&#xff0c;简称DCN。 DCN模型是Wide&Deep的改进版本&#xff0c;其中Deep部分的设计思路与Wide&Deep没有发生本质的变化…

asp.net基于web的校园美食派送配送系统

1&#xff0e;系统登录&#xff1a;系统登录是用户访问系统的路口&#xff0c;设计了系统登录界面&#xff0c;包括用户名、密码和验证码&#xff0c;然后对登录进来的用户判断身份信息&#xff0c;判断是管理员用户还是普通用户。 2&#xff0e;系统用户管理&#xff1a;不管是…

OpenHarmony JS项目开发流程

一、配置OpenHarmony开发环境 1.1软件需求 1&#xff09;下载并安装好DevEco Studio 2.1 Release及以上版本&#xff0c;下载链接&#xff1a;https://developer.harmonyos.com/cn/develop/deveco-studio#download 2&#xff09;获取OpenHarmony SDK包并解压&#xff0c;下载…

学历不仅是敲门砖,也是我下不来的高台,更是孔乙己脱不下的长衫

学历不仅是敲门砖&#xff0c;也是我下不来的高台&#xff0c;更是孔乙己脱不下的长衫 鲁迅《孔乙己》是一篇具有深刻思想和感人情感的短篇小说&#xff0c;通过酒肆里的故事反映社会的残酷和人性的悲哀&#xff1b; 故事中的孔乙己是一个身世不明、生活贫困的酒鬼&#xff0c…

OpenCV学习小记

OpenCV学习小记 &#x1f388;&#x1f388;记在最前&#x1f388;&#x1f388;图像处理的基本操作✨读取图像✨显示图像✨保存图像✨获取图像属性 &#x1f388;&#x1f388;像素的操作✨像素&#x1f514;获取像素的BGR值&#x1f514;修改像素的BGR值 ✨使用NumPy模块操作…