Java每日一题(三道同一类型的题)

 前言

本文一共有三道题:1.两数之和   2.三数之和  3. 四数之和

为什么把这三道题放一起呢,因为三数之和是可以根据两数之和进行推导,四数之和可以根据三数之和进行推导。

 两数之和

思路分析:

我的思路: 1.排序  2.使用左右指针 3.处理细节问题

               先让数组有序,这样就可以使用左右指针的单调性实现了,如果sum比target大我就右指针移动使其sum变小,同理移动右指针。

                3.存在的细节问题,就是去重 和 不要 "漏" 就是找到一种情况后继续寻找 不要 找到 - 2 和 2 而结束 忽略了 后面的 - 1 和 1。去重的主要思想:找到符合条件的左指针跳过前面相同的元素,右指针跳过后面相同的元素 

public List<List<Integer>> twoSum(int[] nums, int target) {
        List<List<Integer>> result = new ArrayList<>();
        //1.排序
        Arrays.sort(nums);
        //2.双指针
        int n = nums.length;
        for (int i = 0; i < n; i++) {
            if (i > 0 && nums[i] == nums[i - 1])continue;//如果当前数与前一个数重复,则跳过
            int start = i, end = n - 1;
            while (start < end){
                long sum = (long) nums[start] + nums[end];
                if (sum > target) end--;
                else if (sum < target) start++;
                else {
                    result.add(new ArrayList<>(Arrays.asList(nums[start],nums[end])));
                    //去重 左指针 a
                    while(start < end && nums[start] == nums[start + 1])start++;
                    //去重 右指针 a
                    while(start < end && nums[end] == nums[end - 1]) end--;
                    //不漏
                    start++;end--;
                }
            }
        }
        return result;
    }

三数之和 

思路分析 :

我的思路:

         1.排序

         2.固定一个数,然后将问题转换为俩个数之和为固定数的相反数

         3.后面利用"双指针算法"快速找到两个的和等于-a即可

         4.处理这个题的细节问题

                1.去重

                        找到一个结果之后,将left和right指针要跳过前面的重复元素

                        当使用完一次固定数,需要跳过相同的固定数

                2.不漏

                        找到一个数之后,不要"停",防止 - 2 和 2后面有 -1 和 1

public List<List<Integer>> threeSum(int[] nums) {
        Arrays.sort(nums);
        List<List<Integer>> result = new ArrayList<>();
        //通过i 判断俩数之和为nums[i]的相反数
        for (int i = 0; i < nums.length - 2; i++) {
            if (i > 0 && nums[i] == nums[i - 1]){
                continue;
            }
            if (nums[i] > 0) return result;
            int findNumber = -nums[i];
            int start = i + 1 ,end = nums.length - 1;
            //不等于查找数进行处理
            while(start < end) {
                if (nums[start] + nums[end] == findNumber) {
                    result.add(Arrays.asList(nums[i], nums[start], nums[end]));
                    //找到结果才需要判断重复元素
                    while (start < end && nums[start] == nums[start + 1]) start++; // 跳过重复元素
                    while (start < end && nums[end] == nums[end - 1]) end--; // 跳过重复元素
                    //不漏 找到一个继续去找下一个三元组
                    start++;
                    end--;
                }
                else if (nums[start] + nums[end] > findNumber) end--;
                    else  start++;
            }
        }
        return result;
    }

四数之和

思路分析:

 我的思路:

         1.排序

         2.固定一个数转化为三数之和为 target - nums[i] 的问题

         3.在固定一个数 转换为两数之和为target - nums[i] - nums[j] 使用双指针进行处理

         4.处理这个题的细节问题

                1.去重

                        去重 (固定数 a) : 最外层循环后跳过 i 和前面相同的数

                        去重 (固定数 b) : 最外层循环后跳过 j 和前面相同的数

                        去重 (左指针 c) : 和两数之和相同

                        去重 (右指针 d) : 和两数之和相同

                2.不漏

                        找到一个数之后,不要"停",防止 - 2 和 2后面有 -1 和 1

public List<List<Integer>> fourSum(int[] nums, int target) {
        List<List<Integer>> result = new ArrayList<>();
        Arrays.sort(nums);
        int n = nums.length;
        //固定一个数转化为三数之和为 target - nums[i] 的问题
        for (int i = 0; i < n - 3; i++) {
            //去重(固定数 a)
            if (i > 0 && nums[i] == nums[i - 1]) continue;//如果当前数与前一个数重复,则跳过
            //在固定一个数 转换为两数之和为target - nums[i] - nums[j] 使用双指针进行处理
            for (int j = i + 1; j < n - 2; j++) {
                //加法操作转换为long防止溢出
                long findNumber = (long)target - nums[i] - nums[j];
                //去重 (固定数 b)
                if (j > i + 1 && nums[j] == nums[j - 1]) continue;// 如果当前数与前一个数重复,则跳过
                int start = j + 1,end = n - 1;
                //双指针进行处理
                while(start < end){//双指针的终止条件
                    if (nums[start] + nums[end] == findNumber){//相等存入集合并且让c d去重
                        result.add(new ArrayList<>(Arrays.asList(nums[i],nums[j],nums[start],nums[end])));
                        //去重 (左指针 c)
                        while (start < end && nums[start] == nums[start + 1]) start++;
                        //去重 (右指针 d)
                        while (start < end && nums[end] == nums[end - 1]) end--;
                        //防漏
                        start++;end--;//继续寻找
                    } else if (nums[start] + nums[end] > findNumber) {//大于 右指针 d 移动让sum更小
                        end--;
                    }else start++;//小于 左指针移动 让sum更大
                }
            }
        }
        return result;
    }

 总结

当解决编程问题时,经常会遇到一系列相关的问题,这些问题往往呈现出渐进式的难度,通过逐步解决这些问题,我们可以逐渐提升自己的编程能力。这种方法也被称为渐进式学习,它的核心思想是在解决一个问题的基础上,逐步构建对更复杂问题的理解和解决能力。

让我们以三道算法题为例来说明这个过程:

  1. 两数之和(Two Sum):给定一个整数数组和一个目标值,找出数组中和为目标值的两个数,并返回它们的下标。这是一个较为简单的问题,可以使用哈希表来解决。

  2. 三数之和(Three Sum):给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0。找出所有满足条件且不重复的三元组。这个问题相比第一道题更复杂一些,可以使用双指针法来解决。

  3. 四数之和(Four Sum):给定一个包含 n 个整数的数组 nums 和一个目标值 target,判断 nums 中是否存在四个元素 a,b,c 和 d,使得 a + b + c + d 的值与 target 相等?找出所有满足条件且不重复的四元组。这是第一道和第二道题目的进阶版,需要更复杂的解决方法。

在解决这三道题目的过程中,我们可以使用"举一反三"的思想。也就是说,我们在解决第一道题目时,可以思考如何将解决方法应用到更复杂的问题中;而在解决第二道题目时,我们可以利用第一道题目的解决方法作为辅助工具,进一步扩展解决的思路;最后,当我们面对第三道题目时,我们可以调用第二道题目的解决方法,并根据需求进行适当的修改和拓展,以解决更复杂的问题。

在实际编程中,我们可以将每道题目的解决方法封装成一个独立的函数或方法。这样,在解决下一道题目时,我们可以直接调用前一道题目的解决方法,提高代码的复用性和可读性。

通过这种渐进式学习的方法,我们可以逐步提升自己的编程能力,并且更好地理解和解决各种复杂的问题。希望这个方法对你有所帮助,也欢迎你在博客中分享你的学习心得和经验,共同进步成长!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

人工智能——深度学习

4. 深度学习 4.1. 概念 深度学习是一种机器学习的分支&#xff0c;旨在通过构建和训练多层神经网络模型来实现数据的高级特征表达和复杂模式识别。与传统机器学习算法相比&#xff0c;深度学习具有以下特点&#xff1a; 多层表示学习&#xff1a;深度学习使用深层神经网络&a…

Java后端常见场景业务问题

目录 单点登录如何实现权限认证如何实现上传数据的安全性如何保证订单表每天新增500W数据,分库分表的方案应该如何设计?订单表每天新增500W数据,分库分表的方案应该如何设计?*********************项目日志如何采集已经上线的bug如何排查如何快速定位系统瓶颈单点登录如何实…

Golang使用PGO优化程序性能

文章目录 参考文章PGO是什么使用PGO的好处PGO做了什么热函数内联什么是内联内联的好处Go默认的内联策略PGO的热函数内联 去虚拟化调用指令高速缓存 PGO有什么缺点可执行程序变大构建时间变长 PGO怎么使用典型的工作流程收集CPU配置文件生产环境启动PGO代码改动重新生成CPU配置文…

基于Whisper语音识别的实时视频字幕生成 (一): 流式显示视频帧和音频帧

Whishow Whistream&#xff08;微流&#xff09;是基于Whisper语音识别的的在线字幕生成工具&#xff0c;支持rtsp/rtmp/mp4等视频流在线语音识别 1. whishow介绍 whishow&#xff08;微秀&#xff09;是在线音视频流播放python实现&#xff0c;支持rtsp/rtmp/mp4等输入&…

人工智能——大语言模型

5. 大语言模型 5.1. 语言模型历史 20世纪90年代以前的语言模型都是基于语法分析这种方法&#xff0c;效果一直不佳。到了20世纪90年代&#xff0c;采用统计学方法分析语言&#xff0c;取得了重大进展。但是在庞大而复杂的语言信息上&#xff0c;基于传统统计的因为计算量巨大…

【Docker】Docker概述及引擎

一、docker概述 DevOps DevOps是一种执行标准&#xff08;思想&#xff09;&#xff0c;主要用于促进开发、测试与运维的整合 容器与虚拟机的区别 最大的区别是&#xff0c;虚拟机中存在独立的硬件系统与操作系统&#xff0c;容器中全部是共享的宿主机中的操作系统与硬件系…

[dvwa] sql injection(Blind)

blind 0x01 low 1’ and length(version()) 6 # syntax: substr(string , from<start from 1>, cut length) 1’ and substr(version(),1,1) ‘5’ # 1’ and substr(version(),2,1) ‘.’ # 1’ and substr(version(),3,1) ‘7’ # 1’ and substr(version(),4,…

酷写写靠谱不 #知识分享#媒体

酷写写是一个值得推荐的论文写作工具&#xff0c;它不仅靠谱而且非常好用。在如今这个信息爆炸的时代&#xff0c;学术界对于论文的要求越来越严格&#xff0c;论文必须具有独创性和高质量才能获得认可。而酷写写的出现&#xff0c;为广大学生和学者提供了一个便捷、高效的写作…

SuperMap三维复杂模型建模之3D极坐标建模——原理篇

作者&#xff1a;超图研究院技术支持中心-于丁 随着SuperMap iDesktop 10i(2021) V10.2.1的上线发布&#xff0c;为进一步拓展全空间数据模型及其分析计算能力&#xff0c;一个新功能“3D极坐标建模”也随着该版本悄然上线。 3D极坐标建模功能实现根据UV参数和数学表达式&…

【Python系列】Jupyter Notebook 中执行 Shell 脚本的方法

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

【随笔】Git 高级篇 -- 项目里程碑 git tag(二十)

&#x1f48c; 所属专栏&#xff1a;【Git】 &#x1f600; 作  者&#xff1a;我是夜阑的狗&#x1f436; &#x1f680; 个人简介&#xff1a;一个正在努力学技术的CV工程师&#xff0c;专注基础和实战分享 &#xff0c;欢迎咨询&#xff01; &#x1f496; 欢迎大…

绝地求生:杜卡迪系列活动集合 各种活动送不停

今天更新已结束&#xff0c;速度上好开冲&#xff01; 春季签到活动&#xff08;第1轮&#xff09; 活动时间 4月9日 - 4月22日 12&#xff1a;00 任务要求 每日登录游戏&#xff08;每日上午10点刷新&#xff09; 活动期间全勤可获得奖励如下 杜卡迪物资箱 (x1) 杜卡迪活…

【C++杂货铺】详解 stack 和 queue

&#x1f308;前言&#x1f308; 欢迎收看本期【C杂货铺】&#xff0c;本期内容将讲解CSTL中stack和queue的内容&#xff0c;其中包含了stack &#xff0c; queue&#xff0c;priority_queue是什么&#xff0c;怎么使用以及模拟实现这些容器。 此外&#xff0c;还将将讲解设计模…

【迅为iMX6Q】开发板 Linux version 6.6.3 SD卡 启动

开发环境 win10 64位 VMware Workstation Pro 16 ubuntu 20.04 【迅为imx6q】开发板&#xff0c; 2G DDR RAM linux-imx 下载 使用 NXP 官方提供的 linux-imx&#xff0c;代码地址为&#xff1a; https://github.com/nxp-imx/linux-imx 使用 git 下载 linux-imx&#xff…

STM32学习和实践笔记(6):自己进行时钟配置的思路

在《STM32学习和实践笔记&#xff08;4&#xff09;: 分析和理解GPIO_InitTypeDef GPIO_InitStructure (d)-CSDN博客》 中&#xff0c;我了解到&#xff0c;在程序执行我们写的main函数之前&#xff0c;实际上先执行了一个汇编语言所写的启动文件&#xff0c;以完成相应的初始…

python爬虫-----爬虫解析—xpath(第十八天)

&#x1f388;&#x1f388;作者主页&#xff1a; 喔的嘛呀&#x1f388;&#x1f388; &#x1f388;&#x1f388;所属专栏&#xff1a;python爬虫学习&#x1f388;&#x1f388; ✨✨谢谢大家捧场&#xff0c;祝屏幕前的小伙伴们每天都有好运相伴左右&#xff0c;一定要天天…

Day:004(4) | Python爬虫:高效数据抓取的编程技术(数据解析)

XPath工具 浏览器-元素-CtrlF 浏览器-控制台- $x(表达式) Xpath helper (安装包需要科学上网) 问题 使用离线安装包 出现 程序包无效 解决方案 使用修改安装包的后缀名为 rar&#xff0c;解压文件到一个文件夹&#xff0c;再用 加载文件夹的方式安装即可 安装 python若使用…

功能测试_验证qq账号的合法性

案例&#xff1a;验证qq账号的合法性&#xff08;要求&#xff1a;6-10位的自然数&#xff09; 使用等价类设计用例案例&#xff1a; 步骤&#xff1a; 1:明确需求&#xff1a;qq账号的合法性 2:划分等价类&#xff1a;有效等价类、有效取值、无效等价类、无效取值 3&…

风电场智能化转型基于ARM工控机的HDMI数据实时监控显示

全球能源结构不断调整的大背景下&#xff0c;智能电网、太阳能发电、风能发电等清洁能源领域正经历着一场由技术创新引领的深刻变革。在这场变革中&#xff0c;ARM架构的工控机凭借其出色的性能、低功耗及高度可定制化的特点&#xff0c;正在成为能源管理系统的核心组件&#x…