算法-双指针-秋招算法冲刺

news2024/9/20 22:56:39

秋招冲刺算法

双指针

数组划分,数组分块

常⻅的双指针有两种形式,⼀种是对撞指针,⼀种是左右指针。

快慢指针

  • 基本思想:使用两个移动速度不同的指针在数组或链表等序列结构上移动。
  • 通常处理结构类型:环形链表或数组。

image-20230613154638391

int slow = 0,fast = 0;
while(fast < length){//相当于right走了一遍,查到分类1的就放到前面。而结束的时候刚好left的下标就是最后一个分类1
    //right指向分类1的情况
    if(arr[right] != 分类2){
        swap(arr[left],arr[right]);
        left++;
    }
    right++;
}

1.移动零

https://leetcode.cn/problems/move-zeroes/

class Solution {
    //分类1 :>0的数
    //分类2 :==0的数
    public void moveZeroes(int[] nums) {
        int left = 0;
        int right =0;
        while(right<nums.length){//这里的分类1 就是>0的数
            if(nums[right]!=0){//当right指向 >0 的数的时候,进行交换
                swap(nums,left,right);
                left++;
            }
            right++;
        }
    }
    public void swap(int[] nums,int left,int right){
        int temp = nums[left];
        nums[left] = nums[right];
        nums[right]  = temp;
    }
}

2.快排核心代码

public void pointerQuicklySort(int[] arr) {
    pointerQuicklySort(arr, 0, arr.length - 1);
}
private void pointerQuicklySort(int[] arr, int left, int right) {
    if (right<=left) return ;
    int root = pointer(arr, left, right);
    pointerQuicklySort(arr, root + 1, right);
    pointerQuicklySort(arr, left, root - 1);
}
private int pointer(int[] arr, int left, int right) {
    int slow = left + 1;//慢指针
    while(fast<=right){
        if(arr[fast]<arr[left]){
            swap(nums,fast,slow);
        	slow++;
        }
        fast++;
    }
    swap(arr, left, slow - 1);//因为到最后slow在带交换的位置上,这位置上的值是大于arr[left]的
    return slow-1;//注意这里也需要slow-1,因为slow位置是基应处位置的右边
}

3.复写零

https://leetcode.cn/problems/duplicate-zeros/

解法:双指针算法,现根据“异地”操作,再优化成“就地”操作。

class Solution {
    public void duplicateZeros(int[] arr) {
//1.定位 fast和slow位置
//都从-1开始,让最后的结果互相对应,fast刚好为slow位置匹配的最后一个数字
        int fast = -1, slow = -1;
        while (fast < arr.length-1) {
            if (arr[slow+1] == 0) {
                slow++;
                fast += 2;
            } else {
                slow++;
                fast++;
            }
        }
//1.5特殊情况判断
//如果这时候slow==0,那么fast位置就会为 arr.length。对这种情况进行处理
        if(fast==arr.length){
            arr[fast-1]=0;
            slow--;
            fast-=2;
        }
//2.开始复写
        while(slow>=0){
            if(arr[slow]==0){
                arr[fast]=0;
                arr[fast-1]=0;
                fast-=2;
                slow--;
            }else{
                arr[fast]=arr[slow];
                fast--;
                slow--;
            }
        }
    }
}

4.快乐数

https://leetcode.cn/problems/happy-number/

按照这个题目走一定会形成环的:int 能形成的最大的数 也不超过 81*10=810 ,一共就有那么多数,但是这种会一直循环无数次,所以一定会产生环,即便这个环只有一个值。

思考方向:判断链表是否有环,使用快慢指针

class Solution {
    public boolean isHappy(int n) {
        int fast = n;int slow = n;
        fast = math(fast);
        while(fast!=slow){
            fast = math(fast);
            fast = math(fast);
            slow = math(slow);
        }
        if(slow==1){
            return true;
        }else{
            return false;
        }
    }
    //写一个函:它每个位置上的数字的平方和
    public int math(int n){
        int ret = 0;
        while(n>0){
            int a = n%10;
            ret+=a*a;
            n/=10;
        }
        return ret;
    }
}

左右针

  • 基本思想:从两端向中间移动。⼀个指针从最左端开始,另⼀个从最右端开始,然后逐渐往中间逼近。

  • 通常处理结构类型:⼀般⽤于顺序结构中,也称左右指针。

1.盛最多水的容器

https://leetcode.cn/problems/container-with-most-water/

image-20230616214122294

import java.util.*;
class Solution {
    public int maxArea(int[] arr) {
        int left = 0,right = arr.length-1;
        int max = 0;
        while(right>left){
            max = Math.max(eara(arr,left,right),max);
            if(arr[left]>arr[right]){
                right--;
            }else{
                left++;
            }
        }
        return max;
    }
    public int eara(int arr[],int left,int right){
        int w = right-left;
        int h = arr[right]<arr[left]?arr[right]:arr[left];
        return w*h;
    }
}

2.剑指 Offer 57. 和为s的两个数字

https://leetcode.cn/problems/he-wei-sde-liang-ge-shu-zi-lcof/

3.三数之和

https://leetcode.cn/problems/3sum/

class Solution {
    //大致思路:固定一个数,剩余的进行 左右指针找
    public List<List<Integer>> threeSum(int[] nums) {
        int left,right;
        List<List<Integer>> ret = new ArrayList<>();
        //1.先进行排序
        Arrays.sort(nums);
        //2.固定一个数
        for(int i=0;i<nums.length;i++){
            //当固定的数 为正数的时候,后面的就没必要判断了
            //因为 无论多少个正数想加都还是正数
            if(nums[i]>0){break;}
 
            left = i+1;
            right = nums.length-1;
            while(right>left){
                if(nums[left]+nums[right]+nums[i] == 0){
                    //System.out.println(nums[left]+" "+nums[right]+" "+nums[i]);
                    List<Integer> list = new ArrayList<>();
                    list.add(nums[i]);
                    list.add(nums[left]);
                    list.add(nums[right]);
                    ret.add(list);
                    left++;right--;
                    //找到之后要进行去重
                    //找到之后:重复的都是复合条件的,已经符合的数直接删掉即可
                    //左右去重:[0,0,0,0]
                    while(right>left && nums[right] == nums[right+1]){
                        right--;
                    }
                    while(right>left && nums[left] == nums[left-1]){
                        left++;
                    }
                }else if(nums[left]+nums[right]+nums[i] > 0){
                    right--;
                }else{
                    left++;
                }
            }
            //根去重:[-1,-1,0,1,2]
            while(i<nums.length-1&&nums[i+1] == nums[i]){
                i++;
            }
        }
        return ret;
    }
}

4.四数之和

https://leetcode.cn/problems/4sum/

和三数想加,思路一样,重点还是 细节把控:重复值的删除

class Solution {
    public List<List<Integer>> fourSum(int[] nums, int target) {
        Arrays.sort(nums);
        List<List<Integer>> ret = new LinkedList<>();
        int left,right;
        for(int i=0;i<nums.length-1;i++){
            for(int j=i+1;j<nums.length-1;j++){
                left = j+1;
                right = nums.length-1;
                while(right>left){
                    //注意int 越界的情况 还强制转换成 long
                    if((long)nums[i]+nums[j]+nums[left]+nums[right] == (long)target){
                        List<Integer> list = new LinkedList<>();
                        list.add(nums[i]);
                        list.add(nums[j]);
                        list.add(nums[left]);
                        list.add(nums[right]);
                        ret.add(list);
                        while(right>left && nums[left+1]==nums[left]){
                            left++;
                        }
                        while(right>left && nums[right-1]==nums[right]){
                            right--;
                        }
                        left++;
                        right--;
                    }else if(nums[i]+nums[j]+nums[left]+nums[right] > target){
                        right--;
                    }else{
                        left++;
                    }
                }
                while(j<nums.length-2 && nums[j+1] == nums[j]){j++;}
            }
            while(i<nums.length-2 && nums[i+1] == nums[i]){i++;}
        }
        return ret;
    }
}

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

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

相关文章

redis协议与异步方式学习笔记

目录 1 交互方式 pipline2 广播机制2.1 概念演示2.2 使用场景 3 redis事物3.1 概念3.2 使用场景3.3 解决的问题3.3.1 背景&#xff1a;多线程竞争出现问题3.3.2 事务3.3.3 安全性事务 3.4两种类型的“事务”3.4.1 watch ... multi exec3.4.2 lua 脚本实现“原子”执行&#xff…

2023-01-11 LightDB高可用常用操作-管理.md

LightDB-高可用常用操作-管理篇 安装环境 操作系统&#xff1a;centos7 服务器IP:1.192.168.121.112 (主)2.192.168.121.113 (从)3.192.168.121.114 (哨兵-可选) 主库重启操作 1.先停止备库的keepalived,在root用户下执行 # 1.获得备库keepalived进 程pid [rootlocalhost ~]#…

闪亮登场!在树莓派上点亮LED灯的简单详细方法

文章目录 树莓派开发与STM32开发的比较原理图以及树莓派引脚展示点灯步骤读取树莓派布局 树莓派开发与STM32开发的比较 树莓派和STM32都是常用的嵌入式设备&#xff0c;都可以使用GPIO来控制LED灯。它们的点灯方式和使用的编程语言以及开发环境略有不同: 相同点&#xff1a; 控…

第五节 Hacker 登录界面

登录框用户界面 1. Entry 输入框 Tkinter中的Entry组件是一个单行文本输入框&#xff0c;用于接收用户在GUI应用程序中输入的信息。它可以被设置为只读或可编辑状态&#xff0c;可以设置输入的文本格式及长度限制等。当用户输入完成后&#xff0c;可以通过调用Entry组件的get(…

软件工程——第3章需求分析知识点整理

本专栏是博主个人笔记&#xff0c;主要目的是利用碎片化的时间来记忆软工知识点&#xff0c;特此声明&#xff01; 文章目录 1.需求分析的基本任务&#xff1f; 2.在需求分析阶段结束前&#xff0c;系统分析员应该做什么&#xff1f; 3.对软件系统有哪些综合要求&#xff1f…

基于Java农家乐信息平台设计实现(源码+lw+部署文档+讲解等)

博主介绍&#xff1a; ✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战 ✌ &#x1f345; 文末获取源码联系 &#x1f345; &#x1f447;&#x1f3fb; 精…

渣土车密闭运输识别算法 yolov7

渣土车密闭运输识别系统通过pythonyolov7网络模型技术&#xff0c;渣土车密闭运输识别算法对渣土车的密闭运输情况进行实时监测&#xff0c;检测到有未密闭的渣土车进入工地区域或者行驶在道路上时&#xff0c;渣土车密闭运输识别算法将自动发出警报提示现场管理人员及时采取措…

【openGauss数据库安全策略配置】--略有小成

【openGauss数据库安全策略配置】--略有小成 &#x1f53b; 一、openGauss数据库安全策略&#x1f530; 1.1 账户安全策略⛳ 1.1.1 自动锁定和解锁帐户&#x1f4a7; 1.1.1.1 配置failed_login_attempts参数&#x1f4a7; 1.1.1.2 配置password_lock_time参数 ⛳ 1.1.2 手动锁定…

MySQL实战解析底层---count(*)这么慢,该怎么办

目录 前言 count(*)的实现方式 用缓存系统保存计数 在数据库保存计数 不同的count用法 前言 在开发系统的时候&#xff0c;你可能经常需要计算一个表的行数&#xff0c;比如一个交易系统的所有变更记录总数这时候你可能会想&#xff0c;一条select count(*) fromt 语句不就…

Spring Boot 如何使用 @ExceptionHandler 注解处理异常消息

Spring Boot 如何使用 ExceptionHandler 注解处理异常消息 在 Spring Boot 应用程序中&#xff0c;异常处理是非常重要的一部分。当应用程序出现异常时&#xff0c;我们需要能够捕获和处理这些异常&#xff0c;并向用户提供有用的错误消息。在 Spring Boot 中&#xff0c;可以…

规则引擎--规则逻辑形如“1 (2 | 3)“的抽象

目录 规则下逻辑表达和条件的抽象表达逻辑的编码和抽象 规则&规则集合条件操作符规则规则执行表达式遍历进行操作符计算添加具体条件的执行 规则执行完成后得到最后的结果 规则下逻辑表达和条件的抽象 对于任何一个规则&#xff0c;包括多个条件&#xff0c;都可以抽象成如…

抽象确实JavaScript

看完上篇的添加事件&#xff0c;我想肯定有一万个黑马在奔腾 明明是照着添加宾语的公式来的&#xff0c;为什么会有报错&#xff1f; 事件不是说可以随便设置吗&#xff1f;但是会出问题 this又是什么关键词&#xff1f; value是啥&#xff1f; 围绕这三大疑问&#xff0c…

LED显示产业如何突破芯片短板

LED显示产业在突破芯片短板方面可以采取以下措施&#xff1a; 研发先进的芯片技术&#xff1a;LED显示芯片的研发是关键。通过投入更多资源和资金&#xff0c;研究机构和企业可以致力于开发更先进、更高效的LED显示芯片技术。这包括改进光电转换效率、提高亮度和色彩表现力等方…

Python基础合集 练习25 (正则表达式)

[0123456789] 普通字符 [0-9]简洁写法 在正则表达式中所有字符类型都有对应的编码 在匹配大写英文字母时,应该是 “”" [a-zA-Z]或[A-Za-z] “”" 元字符… … (.[0-9]{1,3}){3}进行重复三次操作 ^\d{9}$ 使用^和$匹配开始和结束位置,\d表示匹配数字,{9}表示…

【C++篇】C++的动态分配内存

友情链接&#xff1a;C/C系列系统学习目录 知识点内容正确性以C Primer&#xff08;中文版第五版&#xff09;、C Primer Plus&#xff08;中文版第六版&#xff09;为标准&#xff0c;同时参考其它各类书籍、优质文章等&#xff0c;总结归纳出个人认为较有逻辑的整体框架&…

处理 Python 3.11 弃用的 PySys_SetPath 和 Py_SetProgramName

在C调用matplotlibcpp.h画图时报错&#xff0c;使用的python版本是3.11.3版本。 解决方案&#xff1a;不重要的话&#xff0c;注释该行代码即可。 Python 3.11 弃用 PySys_SetPath 和 Py_SetProgramName。这 PyConfig API 取代了这些功能和其他功能。此提交使用 PyConfig API …

深入理解深度学习——GPT(Generative Pre-Trained Transformer):基础知识

分类目录&#xff1a;《深入理解深度学习》总目录 《深入理解深度学习——Transformer》系列文章介绍了Transformer&#xff0c;该模型最初被用于机器翻译任务&#xff0c;其出色表现引起了学术界的极大兴趣&#xff0c;其优异的特征提取与语义抽象能力得到了学者的广泛认可&am…

企业做seo有什么好处?SEO 为您的企业带来的 10 大主要优势?

如果您希望建立长期的品牌知名度、扩大目标受众并赚取更多收入&#xff0c;那么搜索引擎优化至关重要。让我们看看 SEO 为您的企业带来的 10 大好处&#xff0c;以及如何实现它们。 1. 它提高了你的可信度 在搜索引擎结果页面上排名靠前的网站通常被搜索引擎认为是高质量和值得…

【unity细节】—怎么将unity编译时和运行时的功能隔开

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! 本文由 秩沅 原创 收录于专栏&#xff1a;unity细节和bug ⭐怎么将unity编译时和运行时的功能隔开的问题⭐ 文章目录 ⭐怎么将unity编译时和…

TCP的十个核心机制

目录 前言一 到 三四. 滑动窗口五. 流量控制六. 拥塞控制七. 延时应答八. 捎带应答九. 面向字节流十. 异常处理总结 前言 TCP协议是传输层的重点协议, 负责将数据从发送端传输到接收端. TCP协议是传输控制协议, 顾名思义也就是对数据的传输进行控制的协议. TCP 协议有很多, 我…