当前位置: 首页 > news >正文

算法刷题第三天:双指针--2

目录

一,移动零

思路及解法

复杂度分析

二,两数之和 II - 输入有序数组

1,二分查找

 复杂度分析

2,双指针:

复杂度分析

一,移动零

283. 移动零 - 力扣(LeetCode)https://leetcode.cn/problems/move-zeroes/

思路及解法

使用双指针,左指针指向当前已经处理好的序列的尾部,右指针指向待处理序列的头部。

右指针不断向右移动,每次右指针指向非零数,则将左右指针对应的数交换,同时左指针右移。

注意到以下性质:

左指针左边均为非零数;

右指针左边直到左指针处均为零。

因此每次交换,都是将左指针的零与右指针的非零数交换,且非零数的相对顺序并未改变。

class Solution {
public:
    void moveZeroes(vector<int>& nums) {
        int n = nums.size(), left = 0, right = 0;
        while (right < n) {
            if (nums[right]) {
                swap(nums[left], nums[right]);
                left++;
            }
            right++;
        }
    }
};

复杂度分析

  • 时间复杂度:O(n),其中 nn 为序列长度。每个位置至多被遍历两次。

  • 空间复杂度:O(1)。只需要常数的空间存放若干变量。

二,两数之和 II - 输入有序数组

167. 两数之和 II - 输入有序数组 - 力扣(LeetCode)https://leetcode.cn/problems/two-sum-ii-input-array-is-sorted/

之前我们做过这样的一道题,要么直接暴力求解,要么用哈希表去找对应的解。

1,二分查找

在数组中找到两个数,使得它们的和等于目标值,可以首先固定第一个数,然后寻找第二个数,第二个数等于目标值减去第一个数的差。利用数组的有序性质,可以通过二分查找的方法寻找第二个数。为了避免重复寻找,在寻找第二个数时,只在第一个数的右侧寻找。

class Solution {
public:
    vector<int> twoSum(vector<int>& numbers, int target) {
        for (int i = 0; i < numbers.size(); ++i) {
            int low = i + 1, high = numbers.size() - 1;
            while (low <= high) {
                int mid = (high - low) / 2 + low;
                if (numbers[mid] == target - numbers[i]) {
                    return {i + 1, mid + 1};
                } else if (numbers[mid] > target - numbers[i]) {
                    high = mid - 1;
                } else {
                    low = mid + 1;
                }
            }
        }
        return {-1, -1};
    }
};

 复杂度分析


●时间复杂度: O(nlogn), 中n是数组的长度。需要遍历数组一次确定第一个数,时间复杂渡
是O(n),寻找第二个数使用二分查找,时间复杂度是O(logn),因此总时间复杂渡是O(n logn)
●空间复杂度: 0(1)。.

2,双指针:

初始时两个指针分别指向第一个元素位置和最后一 个元素的位置。 每次计算两个指针指向的两个元素
之和,并和目标值比较。如果两个元素之和等于目标值,则发现了唯-解。如果两个元素之和小于目
标值,则将左侧指针右移-位。 如果两个元素之和大于目标值,则将右侧指针左移-位。 移动指针之
后,鰒上述操作,直到找到答案。
使用双指针的实质是缩小查找范围。那么会不会把可能的解过滤掉?答案是不会。假设numbers[i] +
numbers[j]= target 是唯一解, 中0≤i< j≤numbers.length - 1。初始时两个指针分别指向下标0
和下标numbers.length - 1,左指针指向的下标小于或等于i,右指针指向的下标大于或等于j。除非
初始时左指针和右指针已经位于下标i和j,否则-定是左指针先到达下标i的位置或者右指针先到
达下标j的位置。
如果左指针先到达下标i的位置,此时右指针还在下标j的右侧,sum> target,因此一定是右指针
左移,左指针不可能移到i的右侧。
如果右指针先到达下标j的位置,此时左指针还在下标i的左侧,sum< target, 因此一定是左指针
右移,右指针不可能移到j的左侧。
由此可见,在整个移动过程中,左指针不可能移到i的右侧,右指针不可能移到j的左侧,因此不会
把可能的解过滤掉。于题目确保有唯一的答案, 因此使用双指针一定可以找到答案。
 

class Solution {
public:
    vector<int> twoSum(vector<int>& numbers, int target) {
        int low = 0, high = numbers.size() - 1;
        while (low < high) {
            int sum = numbers[low] + numbers[high];
            if (sum == target) {
                return {low + 1, high + 1};
            } else if (sum < target) {
                ++low;
            } else {
                --high;
            }
        }
        return {-1, -1};
    }
};

复杂度分析

  • 时间复杂度:O(n),其中 nn 是数组的长度。两个指针移动的总次数最多为 n 次。

  • 空间复杂度:O(1)。

以上部分来自力扣,由本人整理。

相关文章:

  • 技术分享 | 被测项目需求你理解到位了么?
  • 宿主物种丨Jackson告诉你选择二抗的注意事项
  • centos8安装cobbler3.2
  • 网络编程-----socket函数
  • SpringBoot的自动装配进阶
  • 高通平台Android 蓝牙调配置手试和册-- OPP File Transmission Failure
  • STC15单片机内部RAM讲解
  • zemax---Ray Aberration(光线光扇图)
  • Polygon zkEVM Arithmetic状态机
  • 汽车毫米波雷达测试与测量解决方案
  • 网络编程--sockaddr 与 sockaddr_in
  • HashMap底层分析
  • 《工程伦理与学术道德》之《导论》
  • VGLUT 1抗体丨SYSY VGLUT 1抗体化学性质和文献参考
  • 598. 范围求和 II (脑筋急转弯)
  • [译]CSS 居中(Center)方法大合集
  • 【翻译】babel对TC39装饰器草案的实现
  • 2018一半小结一波
  • ECS应用管理最佳实践
  • gf框架之分页模块(五) - 自定义分页
  • JavaScript函数式编程(一)
  • nfs客户端进程变D,延伸linux的lock
  • PHP 小技巧
  • RxJS 实现摩斯密码(Morse) 【内附脑图】
  • SpiderData 2019年2月13日 DApp数据排行榜
  • vue-cli在webpack的配置文件探究
  • 观察者模式实现非直接耦合
  • 买一台 iPhone X,还是创建一家未来的独角兽?
  • 深度学习中的信息论知识详解
  • 算法---两个栈实现一个队列
  • 云栖大讲堂Java基础入门(三)- 阿里巴巴Java开发手册介绍
  • 如何通过报表单元格右键控制报表跳转到不同链接地址 ...
  • ​【已解决】npm install​卡主不动的情况
  • ​520就是要宠粉,你的心头书我买单
  • # 飞书APP集成平台-数字化落地
  • (11)MSP430F5529 定时器B
  • (C语言)求出1,2,5三个数不同个数组合为100的组合个数
  • (C语言)深入理解指针2之野指针与传值与传址与assert断言
  • (附源码)springboot 基于HTML5的个人网页的网站设计与实现 毕业设计 031623
  • (附源码)springboot建达集团公司平台 毕业设计 141538
  • (附源码)ssm基于jsp的在线点餐系统 毕业设计 111016
  • (含react-draggable库以及相关BUG如何解决)固定在左上方某盒子内(如按钮)添加可拖动功能,使用react hook语法实现
  • (切换多语言)vantUI+vue-i18n进行国际化配置及新增没有的语言包
  • (一)Dubbo快速入门、介绍、使用
  • (一)使用Mybatis实现在student数据库中插入一个学生信息
  • (转)ABI是什么
  • (转)平衡树
  • ***测试-HTTP方法
  • .[backups@airmail.cc].faust勒索病毒的最新威胁:如何恢复您的数据?
  • .form文件_SSM框架文件上传篇
  • .NET Core引入性能分析引导优化
  • .NET的数据绑定
  • .net之微信企业号开发(一) 所使用的环境与工具以及准备工作
  • @DataRedisTest测试redis从未如此丝滑
  • @软考考生,这份软考高分攻略你须知道