算法刷题Day1 | 704.二分查找、27.移除元素

目录

  • 0 引言
  • 1 二分查找
    • 1.1 我的解题
    • 1.2 修改后
    • 1.3 总结
  • 2 移除元素
    • 2.1 暴力求解
    • 2.2 双指针法(快慢指针)

请添加图片描述

  • 🙋‍♂️ 作者:海码007
  • 📜 专栏:算法专栏
  • 💥 标题:代码随想录算法训练营第一天 | 704.二分查找、27.移除元素
  • ❣️ 寄语:书到用时方恨少,事非经过不知难!

0 引言

1 二分查找

  • 🎈 文档讲解:https://programmercarl.com/0704.%E4%BA%8C%E5%88%86%E6%9F%A5%E6%89%BE.html
  • 🎈 视频讲解:https://www.bilibili.com/video/BV1fA4y1o715/
  • 🎈 做题状态:有思路但是不清晰,对于循环的判断条件还有区间的变换存在疑惑(这些就是二分法的易错点)

1.1 我的解题

在这里插入图片描述

class Solution {
public:
    int search(vector<int>& nums, int target) {
        int left, right, middle;
        left = 0;
        right = nums.size() - 1;

        if (right == left)
        {
            if (target == nums[left])
            {
                return left;
            }
            else
            {
                return -1;
            }
        }
        
        while (right != left + 1)
        {
            middle = (right - left) / 2 + left;
            cout << "middle = " << middle << endl;
            if (target == nums[middle])
            {
                return middle;
            }
            else if (target > nums[middle])
            {
                left = middle;
            }
            else
            {
                right = middle;
            }
        }
        if (target == nums[left])
        {
            return left;
        }
        else if (target == nums[right])
        {
            return right;
        }
        else
        {
            return -1;
        }
    }
};

分析:没有标准答案简洁,原因是我的 leftright 赋值思路不对(相差1)。我是判断完大小后直接将 leftright 直接赋值 middle 。其实这样不对,因为 middle 的值已经判断过了,不需要再将这个索引值包含在 [left, right]区间中。

1.2 修改后

所以需要将代码进行如下修改,这样可以省去很多其他条件的判断。

class Solution {
public:
   int search(vector<int>& nums, int target) {
       int left, right, middle;
       left = 0;
       right = nums.size() - 1;
       
       while (right >= left)
       {
           middle = (right - left) / 2 + left;
           cout << "middle = " << middle << endl;
           if (target == nums[middle])
           {
               return middle;
           }
           else if (target > nums[middle])
           {
               left = middle + 1;
           }
           else
           {
               right = middle - 1;
           }
       }
       return -1;
   }
};

1.3 总结

看到顺序排列的数组,就可以联想到二分法

二分法易错点:

  1. while循环条件:
    • while(left <= right)
    • while(left < right)
  2. left和right的赋值
    • left = middle+1; right = middle -1;
    • left = middle+1; right = middle;

其实while和left、right的赋值是对应的,

  • 使用while(left <= right)循环时,left = middle+1; right = middle -1;
  • 使用while(left < right)循环时,left = middle+1; right = middle;

因为当 left = middle+1; right = middle -1; left和right都把已经判断过的 middle 索引给排除在区间之外了,所以循环遍历的条件需要包括 left和right ,也就是 左闭右闭 区间。
left = middle+1; right = middle; 可以发现right没有把已经判断过的 middle 索引给排除在区间之外,所以循环条件不需要包括 right 的索引, 也就是 左闭右开 区间。


使用左闭右闭 区间时,最后left和right的结果是什么,当 left等于right的时候(此时middle也等于left和right),还会再执行一次循环,当nums[middle] > target 时 right = middle+1; 也就是说right又往右移动了一位。当nums[middle] <= target 时,right保持原位置不变。
这个特性可以帮助 35.搜索插入位置 这道题理解。

2 移除元素

  • 🎈 文档讲解:https://www.programmercarl.com/
  • 🎈 视频讲解:https://www.bilibili.com/video/BV12A4y1Z7LP/?vd_source=d499e7f3a8e68e2b173b1c6f068b2147
  • 🎈 做题状态:暴力求解没太大难度,快慢指针有需要理解的地方

2.1 暴力求解

一开始用暴力求解,输出答案不对,代码如下。我的的输出结果总是不能把最后一位元素给移除,后面发现遍历时没有遍历到最后一个元素。i < arraySize - 1 判断条件没写好,应该是 i < arraySize 或者 i <= arraySize - 1 。这个易错点以后要注意!!!

在这里插入图片描述


在这里插入图片描述

class Solution {
public:
    int removeElement(vector<int>& nums, int val) {
        int arraySize = nums.size();
        for (int i = 0; i <= arraySize - 1; i++)
        {
            cout << "i = " << i << endl;
            if (nums[i] == val)
            {
                for (int j = i; j < arraySize - 1; j++)
                {
                    nums[j] = nums[j + 1];
                }
                --arraySize;
                --i;
            }
        }
        return arraySize;
    }
};

2.2 双指针法(快慢指针)

快指针:遍历元素值
慢指针:存储新数组的元素值

思路:
遇到等于目标元素的值就跳过,遇到不等于目标元素的值就进行赋值。借助两个索引值(fastIndex和slowIndex)实现,fastIndex遇到目标元素就跳过,此时slowIndex不动,因为目标元素不需要存储。然后遇到不等于目标元素的值就进行赋值。此时是fastIndex在前面探路,然后slowIndex将fastIndex探路得到的结果进行保存。

因为数组的存放是顺序的,所以只有在遇到不等于目标元素的值时,slowIndex才会增加一,然后将非目标值保存下来。

fastIndex从头遍历到尾,所以直接用fastIndex作为遍历元素,条件为 fastIndex < nums.size() 即可。然后当 fastIndex 所指元素不等于目标元素值时,此时将 fastIndex 的值赋值给slowIndex 处。并且slowIndex加一。由此可知 nums 数组中有多少个不等于目标值的元素(个数就是slowIndex)。

class Solution {
public:
    int removeElement(vector<int>& nums, int val) {
        int fastIndex;
        int slowIndex = 0;

        for ( fastIndex = 0; fastIndex < nums.size(); fastIndex++)
        {
            if (nums[fastIndex] != val)
            {
                nums[slowIndex] = nums[fastIndex];
                ++slowIndex;
            }
        }
        
        return slowIndex;
    }
};

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

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

相关文章

Vue.js+SpringBoot开发大病保险管理系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 系统配置维护2.2 系统参保管理2.3 大病保险管理2.4 大病登记管理2.5 保险审核管理 三、系统详细设计3.1 系统整体配置功能设计3.2 大病人员模块设计3.3 大病保险模块设计3.4 大病登记模块设计3.5 保险审核模块设计 四、…

MySQL三种日志

一、undo log&#xff08;回滚日志&#xff09; 1.作用&#xff1a; &#xff08;1&#xff09;保证了事物的原子性 &#xff08;2&#xff09;通过read view和undo log实现mvcc多版本并发控制 2.在事务提交前&#xff0c;记录更新前的数据到undo log里&#xff0c;回滚的时候读…

数据可视化助力林业智能管理

数据可视化是当下科技发展中的一项重要工具&#xff0c;它在各行各业都展现了强大的应用价值。在智慧林业领域&#xff0c;数据可视化更是发挥了独特的作用&#xff0c;为林业管理和生态保护提供了有效的支持和解决方案。下面我就以可视化从业者的角度&#xff0c;来简单聊聊这…

四节点/八节点四边形单元悬臂梁Matlab有限元编程 | 平面单元 | Matlab源码 | 理论文本

专栏导读 作者简介&#xff1a;工学博士&#xff0c;高级工程师&#xff0c;专注于工业软件算法研究本文已收录于专栏&#xff1a;《有限元编程从入门到精通》本专栏旨在提供 1.以案例的形式讲解各类有限元问题的程序实现&#xff0c;并提供所有案例完整源码&#xff1b;2.单元…

关于yolov8的DFL模块(pytorch以及tensorrt)

先看代码 class DFL(nn.Module):"""Integral module of Distribution Focal Loss (DFL).Proposed in Generalized Focal Loss https://ieeexplore.ieee.org/document/9792391"""def __init__(self, c116):"""Initialize a convo…

嵌入式C语言(六)

对齐这个事情在内核中可不是个什么小事&#xff0c;内核中涉及到内存方面的都需要非常的谨慎。 上一篇我们知道了可以通过__attribute__来声明属性&#xff0c;也知道了section这个属性&#xff0c;这篇我们来看看关于内存对齐使用的两个属性–>aligned和packed 地址对齐&…

Altium Designer如何对走线模式进行切换

AD软件提供了比较智能的走线模式切换功能&#xff0c;可以根据个人习惯进行切换&#xff0c;能有效的提高了PCB设计效率。 点击界面右上角系统参数的图标 或者在pcb界面中使用快捷键OP进入到优选项界面&#xff0c;然后选中 PCB Editor-Interactive Routing&#xff0c;在布线…

ubuntu設定QGC獲取pixhawk Mini4(PX4 Mini 4) 的imu信息

ubuntu20.04 QGC使用v4.3.0的版本 飛控pixhawk Mini4 飛控上只使用一條micro USB連接電腦&#xff0c;沒有其他線 安裝命令 sudo apt-get remove modemmanager -y sudo apt install gstreamer1.0-plugins-bad gstreamer1.0-libav gstreamer1.0-gl -y sudo apt install libf…

Vue:纯前端实现文件拖拽上传

先看一下拖拽相关的事件&#xff1a;dragover、dragenter drop和dragleave 。 dragover事件&#xff1a;当被拖动的元素在一个可放置目标上方时&#xff0c;该事件会被触发。 通常&#xff0c;我们会使用event.preventDefault()方法来取消浏览器默认的拖放行为&#xff0c;以便…

amv是什么文件格式?如何播放amv视频?

AMV文件格式源自于中国公司Actions Semiconductor&#xff0c;最初作为其MP4播放器中使用的专有视频格式。产生于数码媒体发展的需求下&#xff0c;AMV格式为小屏幕便携设备提供了一种高度压缩的视频存储方案。 AMV文件格式的主要特性与使用场景 AMV格式以其独特的特性在小尺寸…

复合式统计图绘制方法(7)

复合式统计图绘制方法&#xff08;7&#xff09; 常用的统计图有条形图、柱形图、折线图、曲线图、饼图、环形图、扇形图。 前几类图比较容易绘制&#xff0c;饼图环形图绘制较难。 在统计图的应用方面&#xff0c;有时候有两个关联的统计学的样本值要用统计图来表达&#xff0…

运动想象 (MI) 迁移学习系列 (5) : SSMT

运动想象迁移学习系列:SSMT 0. 引言1. 主要贡献2. 网络结构3. 算法4. 补充4.1 为什么设置一种新的适配器&#xff1f;4.2 动态加权融合机制究竟是干啥的&#xff1f; 5. 实验结果6. 总结欢迎来稿 论文地址&#xff1a;https://link.springer.com/article/10.1007/s11517-024-0…

天府锋巢直播产业基地:直播带岗,成都直播基地奔向产业化

天府锋巢直播产业基地位于成都市天府新区科学城板块&#xff0c;是一座集直播带岗、电商孵化、产业培训、供应链整合等多功能于一体的现代化全域直播产业基地。近年来&#xff0c;随着成都直播产业的蓬勃发展&#xff0c;成都积极响应市场需求&#xff0c;致力于打造出西部地区…

linux进程间通信-共享内存

一、共享内存是什么 在Linux系统中&#xff0c;共享内存是一种IPC&#xff08;进程间通信&#xff09;方式&#xff0c;它可以让多个进程在物理内存中共享一段内存区域。 这种共享内存区域被映射到多个进程的虚拟地址空间中&#xff0c;使得多个进程可以直接访问同一段物理内存…

【Python可视化系列】一文教你绘制雷达图(源码)

这是我的第234篇原创文章。 一、引言 雷达图是以从同一点开始的轴上表示的三个或更多个定量变量的二维图表的形式显示多变量数据的图形方法&#xff0c;也称为蜘蛛图或星形图。雷达图通常用于综合分析多个指标&#xff0c;具有完整&#xff0c;清晰和直观的优点。通常由多个等…

Constrained Iterative LQR 自动驾驶中使用的经典控制算法

Motion planning 运动规划在自动驾驶领域是一个比较有挑战的部分。它既要接受来自上层的行为理解和决策的输出,也要考虑一个包含道路结构和感知所检测到的所有障碍物状态的动态世界模型。最终生成一个满足安全性和可行性约束并且具有理想驾驶体验的轨迹。 通常,motion plann…

遥感影像植被波谱特征总结

植被跟太阳辐射的相互关系有别于其他物质&#xff0c;如裸土、水体等&#xff0c;比如植被的“红边”现象&#xff0c;即在<700nm附近强吸收&#xff0c;>700nm高反射。很多因素影响植被对太阳辐射的吸收和反射&#xff0c;包括波长、水分含量、色素、养分、碳等。 研究…

Kubernetes--ingress实现七层负载

目录 一、传统方式&#xff1a;不借助ingress实现七层代理 二、nginx-ingress 三、使用ingress实现七层代理 四、部署ingrss-nginx及功能 五、样例 1.Ingress-nginx HTTP代理访问 2.Ingress HTTPS代理访问&#xff08;会话卸载层&#xff09; 3.Nginx进行BasicAuth&…

亚马逊店铺解决和预防订单下滑的技巧

1. 保持账号的良好表现。不要销售侵权产品&#xff0c;发货要及时&#xff0c;能有追踪号的就带可查询追踪号&#xff0c;能发FBA的就通过FBA发货。 2. 持续做好产品优化工作&#xff0c;及时留意大环境的变化和平台政策变动。遇到编辑权限受限&#xff0c;可开case咨询或申请…

【数据库】软件测试之MySQL数据库练习题目

有表如下&#xff1a; Student 学生表 SC 成绩表 Course 课程表 Teacher 老师表 每个学生可以学习多门课程&#xff0c;每一个课程都有得分&#xff0c;每一门课程都有老师来教&#xff0c;一个老师可以教多个学生 1、查询姓‘朱’的学生名单 select * from Student whe…