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

news2024/11/28 10:38:38

 前言

本文一共有三道题: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 相等?找出所有满足条件且不重复的四元组。这是第一道和第二道题目的进阶版,需要更复杂的解决方法。

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

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

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

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

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

相关文章

【C++成长记】C++入门 |函数重载、引用、内联函数

&#x1f40c;博主主页&#xff1a;&#x1f40c;​倔强的大蜗牛&#x1f40c;​ &#x1f4da;专栏分类&#xff1a;C❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ 目录 一、函数重载 1、函数重载概念 二、引用 1、引用概念 2、引用特性 3、常引用 4、使用场景 5、…

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

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

秋叶Stable diffusion的创世工具安装-带安装包链接

来自B站up秋葉aaaki&#xff0c;近期发布了Stable Diffusion整合包v4.7版本&#xff0c;一键在本地部署Stable Diffusion&#xff01;&#xff01; 适用于零基础想要使用AI绘画的小伙伴~本整合包支持SDXL&#xff0c;预装多种必须模型。无需安装git、python、cuda等任何内容&am…

wordpress全站开发指南-面向开发者及深度用户(全中文实操)--wordpress中的著名循环

wordpress中的著名循环 首先&#xff0c;在深入研究任何代码之前&#xff0c;我们首先要确保我们有不止一篇博客文章可以工作。因此&#xff0c;我们要去自己的wordpress站点&#xff0c;从侧边栏单机Posts(文章)&#xff0c;进行创建 在执行代码的时候会优先执行single.php如…

【算法刷题】八大排序算法总结(冒泡、选择、插入、二分插入、归并、快速、希尔、堆排序)

文章目录 八大排序算法总结1.冒泡排序2.选择排序3.插入排序4.二分插入排序5.归并排序6.快速排序7.希尔排序8.堆排序 八大排序算法总结 排序排序方法平均情况最好情况最坏情况空间稳定性1冒泡排序O(n2)O(n)O(n2)O(1)稳定2选择排序O(n2)O(n2)O(n2)O(1)不稳定3插入排序O(n2)O(n)O…

生活中的数学 --- 等额本息贷款和等额本金贷款的月供应该怎么算?

等额本息贷款和等额本金贷款的月供应该怎么算&#xff1f; 从一个例子开始&#xff0c;假设我要从银行贷款36万(即&#xff0c;本金)&#xff0c;银行给出的贷款年利率是12%(月利率为年利率除以12)&#xff0c;贷款半年(6个月)&#xff0c;按月还款&#xff0c;分6期还完。 问分…

go websocket

WebSocket 是一种网络协议&#xff0c;建立在 HTTP 协议之上&#xff0c;允许双向通信。WebSocket 协议允许服务器发送数据到客户端&#xff0c;同时也可以让客户端向服务器发送数据。WebSocket 使用 HTTP 协议的升级请求和响应来建立连接。WebSocket 的主要优点在于它可以通过…

【Node.js】短链接

原文链接&#xff1a;Nodejs 第六十二章&#xff08;短链接&#xff09; - 掘金 (juejin.cn) 短链接是一种缩短长网址的方法&#xff0c;将原始的长网址转换为更短的形式。短链接的主要用途之一是在社交媒体平台进行链接分享。由于这些平台对字符数量有限制&#xff0c;长网址可…

Lua热更新(AssetBundle)

AssetBundle 新版本导入ab包报错,则删除其中的Tests文件夹。 给资源分组 打包设置:平台、路径、重复打包清空文件夹、复制到streaming文件夹 建议勾选 建议使用LZ4压缩方式 用来观察文件中的包大小,不常用 参数总结: 这六个只做了解,重要的是上面的

kubesphere部署(apple m1 m2 m3)

背景&#xff1a;使用一个命令kk(KubeKey)同时快速安装 Kubernetes 和 KubeSphere的集成环境&#xff0c;提高效率&#xff0c;减少部署时所花费的精力。这里环境为apple m2 一、KubeSphere简介 KubeSphere 是在 Kubernetes 之上构建的面向云原生应用的分布式操作系统&#x…

YOLOv7全网独家改进: 卷积魔改 | 变形条状卷积,魔改DCNv3二次创新

💡💡💡本文独家改进: 变形条状卷积,DCNv3改进版本,不降低精度的前提下相比较DCNv3大幅度运算速度 💡💡💡强烈推荐:先到先得,paper级创新,直接使用; 💡💡💡创新点:1)去掉DCNv3中的Mask;2)空间域上的双线性插值转改为轴上的线性插值; 💡💡💡…

人工智能——深度学习

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

智过网:报考中级注册安全工程师需要什么条件?

随着社会的快速发展和科技的日新月异&#xff0c;安全生产问题越来越受到人们的关注。中级注册安全工程师作为专业安全管理人才&#xff0c;其职责与角色日益凸显。那么&#xff0c;想要报考中级注册安全工程师&#xff0c;需要满足哪些条件呢&#xff1f; 首先&#xff0c;报考…

lanqiao.125卡片换位(2016年蓝桥杯C/C++省赛C组)

题目&#xff1a; 语法点&#xff1a; 1. unordered_map<string,int> dist; //存储图的不同状态及不同状态对应的步数 2. unordered_map的相关操作&#xff0c;详细见C中的unordered_map用法详解-CSDN博客 dist.count(x) //来寻找x出现的次数 dist.find(x) //来…

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

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

51单片机学习笔记15 LCD12864(带字库)显示屏使用

51单片机学习笔记15 LCD12864&#xff08;带字库&#xff09;显示屏使用 一、LCD12864简介二、管脚定义三、命令1. 功能能设定2. 清屏指令&#xff08;0x01&#xff09;3. 地址归位4. 进入设定点5. 显示状态开关6. 设定CGRAM地址7. 设定DDRAM地址8. 写资料到RAM9. 读出RAM 四、…

Flink WordCount实践

目录 前提条件 基本准备 批处理API实现WordCount 流处理API实现WordCount 数据源是文件 数据源是socket文本流 打包 提交到集群运行 命令行提交作业 Web UI提交作业 上传代码到gitee 前提条件 Windows安装好jdk8、Maven3、IDEA Linux安装好Flink集群&#xff0c;可…

37-代码测试(下):Go语言其他测试类型及IAM测试介绍

。 Go中的两类测试&#xff1a;单元测试和性能测试。 我就来介绍下Go 语言中的其他测试类型&#xff1a;示例测试、TestMain函数、Mock测试、Fake测试等&#xff0c; 示例测试 示例测试以Example开头&#xff0c;没有输入和返回参数&#xff0c;通常保存在example_test.go…

Verilog实现手表计时

实现手表的计时功能&#xff1a; 1.具有start启动信号、pause暂停信号&#xff0c;可以自定义其触发机制。 2.具有时间更改接口&#xff0c;可以更改时、分、秒。 3.输出时、分、秒。 Verilog设计 模块端口定义&#xff1a; module watch1(input wire clk …

mp4转flv怎么转?电脑怎么把视频转成flv?

MP4&#xff08;MPEG-4 Part 14&#xff09;是一种多媒体容器格式&#xff0c;广泛用于包含视频、音频、字幕等多种数据流。MP4因其高度灵活性、压缩效率和兼容性成为视频领域的主流格式&#xff0c;支持范围涵盖从在线视频到移动设备的各类应用场景。 FLV文件格式的多个优点 …