春招Leetcode刷题日记-D3-双指针算法-归并两个有序数组Two Sum

news2024/9/30 1:39:06

D3-双指针算法-归并两个有序数组&&Two Sum

  • 什么是双指针算法
  • 力扣88. 合并两个有序数组
    • 思路
    • 代码
  • 力扣167. 两数之和 II - 输入有序数组
    • 思路
      • 思路一:时间复杂度O(nlogn)算法
      • 代码
      • 思路二:时间复杂度O(n)--双指针算法
      • 代码


什么是双指针算法

1、双指针主要用于遍历数组,两个指针指向不同的元素,从而协同完成任务。也可以延伸到多
个数组的多个指针。
2、若两个指针指向同一数组,遍历方向相同且不会相交,则也称为滑动窗口(两个指针包围的区域即为当前的窗口),经常用于区间搜索。
3、若两个指针指向同一数组,但是遍历方向相反,则可以用来进行搜索,待搜索的数组往往是排好序的。
4、常见的c++指针操作方式:

int x;
int * p1 = &x; // 指针可以被修改,值也可以被修改
const int * p2 = &x; // 指针可以被修改,值不可以被修改(const int)
int * const p3 = &x; // 指针不可以被修改(* const),值可以被修改
const int * const p4 = &x; // 指针不可以被修改,值也不可以被修改
// addition是指针函数,一个返回类型是指针的函数
int* addition(int a, int b) {
	int* sum = new int(a + b);
	return sum;
}
int subtraction(int a, int b) {
	return a - b;
}
int operation(int x, int y, int (*func)(int, int)) {
	return (*func)(x,y);
}
// minus是函数指针,指向函数的指针
int (*minus)(int, int) = subtraction;
int* m = addition(1, 2);
int n = operation(3, *m, minus);

力扣88. 合并两个有序数组

题目链接:88. 合并两个有序数组

在这里插入图片描述
在这里插入图片描述

思路

1、这种题是最基本的套路提,最简单的想法就是:两个指针在两个数组头部进行遍历,每次从两个指针指向的位置处选择小的元素,最后在把没放进去的放进去即可。
2、但需要额外存于一个新数组,空间复杂度不好。
3、所以,采取优化手段,我们两个指针,从两个数组尾部开始遍历,每次选大的加入nums尾部(已经提前开好空间了),这样就可以省下空间

代码

class Solution {
public:
	void merge(vector<int>& nums1, int m, vector<int>& nums2, int n) {
		int i = m - 1, j = n - 1, k = m + n - 1;//分别指向nums1,nums2,k用于在nums1尾部写入新数据
		while (i >= 0 && j >= 0) {
			if (nums2[j] > nums1[i]) {//每次放入大的元素进入nums1尾部
				nums1[k] = nums2[j];
				j--;
			}
			else {
				nums1[k] = nums1[i];
				i--;
			}
			k--;
		}
		while (i >= 0) {//把剩余的放进去
			nums1[k] = nums1[i];
			i--;
			k--;
		}
		while (j >= 0) {
			nums1[k] = nums2[j];
			j--;
			k--;
		}
	}
};

力扣167. 两数之和 II - 输入有序数组

题目链接:167. 两数之和 II - 输入有序数组
在这里插入图片描述
在这里插入图片描述

思路

拿来这个题,最基本的想法就是,我遍历一遍数组,访问到i处时,再遍历一遍数组,查找target-numbers[i]是否在数组中,这是O(n^2)的想法,也是最朴素的想法。

思路一:时间复杂度O(nlogn)算法

1、但是,只要是涉及在有序数组中查找一个数值的问题的时候,一定一定一定不要犹豫!直接选择二分查找!
2、注意读题,答案中下标顺序,一定是递增的,所以从左到右枚举第一个i,再对第二个数二分查找就行,返回下标
3、注意,第二个数的下标,一定不能和第一个数的下标重合,二分查找的时候注意判断。
4、根据题意,最后下标都加一

代码

class Solution {
public:
	vector<int> twoSum(vector<int>& numbers, int target) {//二分查找
		int n = numbers.size();
		for (int i = 0; i < n; i++) {
			int one = numbers[i];
			int two = target - one;
			int pos = find(numbers, 0, n - 1, two, i);
			if (pos != -1) {
				return { i + 1,pos + 1 };
			}
		}
		return {};
	}
	//二分查找参数说明:数组(传入引用,加快效率),左边界,右边界,目标数,第一个数对应的下标
	int find(vector<int>& numbers, int left, int right, int two, int i) {//二分查找
		if (left > right) {
			return -1;
		}
		int mid = (left + right) / 2;
		if (numbers[mid] == two && mid != i) {
			return mid;
		}
		else if (numbers[mid] > two) {
			return find(numbers, left, mid - 1, two, i);
		}
		return find(numbers, mid + 1, right, two, i);
	}
};

思路二:时间复杂度O(n)–双指针算法

1、我们一定要善于具体化搜索过程,根据朴素搜索定义,相当于一个指针i在一个轴上面搜索,另一个指针j在一个轴上面搜索,也就是,可以具体看做在一个二维平面上搜索,每一个(i,j)里面的值,代表着numbers[i]+numbers[j],相当于在一个二维矩阵(已经排好序)中搜索目标值。
2、但是,由于题意,我们只能在上半三角矩阵中搜索,如图中白色框框:
在这里插入图片描述
3、所以,这道题就转化为了,和240. 搜索二维矩阵 II一模一样的思路,从左下角或者右上角开始,在O(n)内即可完成搜索。下面,我将介绍这种搜索的原理
4、因为本体限制,所以只能从右上角在(0,7)处开始搜索,如果不是目标要么大于target,要么小于target

当小于target时,因为在0行中,(0,7)已经是最大了,再想要更大的数,i=0根本满足不了,直接省略0行,i++
当大于target时,因为在7列中,(0,7)已经是最小了,再想要更小的数,j=7根本满足不了,直接省略7列,j - -

5、综上,可以看到,无论 结果是大了还是小了,我们都可以排除掉一行或者一列的搜索空间。经过 n 步以后,就能排除所有的搜索空间,检查完所有的可能性。搜索空间的减小过程如下面动图所示:

请添加图片描述
这也就是,这一类已经全部排序完毕的二维空间中,快速查找的思路。

代码

class Solution {
public:
	vector<int> twoSum(vector<int>& numbers, int target) {//双指针
		int i = 0, j = numbers.size() - 1;
		while (i < j) {
			if (numbers[i] + numbers[j] < target) {
				i++;
			}
			else if (numbers[i] + numbers[j] > target) {
				j--;
			}
			else {
				return { i+1,j+1 };
			}
		}
		return {};
	}
};

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

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

相关文章

c++学习笔记-案例-职工管理系统(哔站-黑马程序员c++教学视频)

目录 一、需求分析 1、职工管理系统需求 2、管理系统中需要实现的功能 3、系统界面如下&#xff1a; 二、代码实现 1、整体结构 2、具体分析 &#xff08;1&#xff09;创建01workerManager.h &#xff08;2&#xff09;创建02worker.h &#xff08;3&#xff09;创建…

秒杀项目的消息推送

目录 一、创建消费者 二、创建订单链路配置 1.定义RabbitMQ配置类 2.创建RabbitmqOrderConfig配置类 三、如何实现RabbitMQ重复投递机制 1.开启发送者消息确认模式 2.消息发送确认 ① 创建ConfirmCallBacker确认模式 ② 创建ReturnCallBack退回模式 3.创建生产者 …

*1创建 Vue3

1、使用 vue-cli 进行创建 &#xff08;1&#xff09;查看 vue-cli 版本,确保 vue-cli 版本在4.5.0上 win键 r &#xff0c;输入cmd ——>输入 vue -V 若低于4.5.0版本&#xff0c;则需要重新安装 npm i -g vue/cli &#xff08;2&#xff09;在桌面创建项目&#xff08;也…

天荒地老修仙功-第七部:Dubbo基本使用与原理详解

文章目录一、基础知识1、分布式基础理论1.1、什么是分布式系统&#xff1f;1.2、发展演变1.2.1 单一应用架构1.2.2 垂直应用架构1.2.4 分布式服务架构1.2.4 流动计算架构1.3、RPC2、dubbo核心概念2.1、简介2.2、基本概念2.2.1 服务提供者&#xff08;Provider&#xff09;2.2.2…

ViewBinding使用入门

ViewBinding 参考资料: 新技术 ViewBinding 最佳实践 & 原理击穿 更多 ViewBinding 的封装思路 1. kotlin-android-extensions(KAE) 的问题 根据Google官方的说法, KAE存在以下问题: 污染全局命名空间不能暴露可空性信息仅支持Kotlin代码 kotlin在1.4.20中 开始废弃这…

广州暨大港澳子弟学校小学部IB探究

作为IB世界会员学校的一员&#xff0c;培养学生成为具有国际情怀的人&#xff0c;承认人类共有的博爱精神&#xff0c;分担守护地球的责任&#xff0c;帮助开创一个更好、更和平的世界而努力是广州暨大港澳子弟学校&#xff08;ASJ&#xff09;的教育使命和目标。 ASJ在PYP项目…

使用Junit框架,提高自动化测试效率

目录 1、Junit 是什么&#xff1f; 2、Junit 的用法 2.1、注解【Test、BeforeEach、BeforeAll、AfterEach、AfterAll】 2.2、断言Assertion类 2.2.1、断言匹配/不匹配 2.2.2、断言结果为真/为假 2.2.3、断言结果为空/不为空 2.3、用例的执行顺序【order注解】 2.4、参…

字节跳动青训营--前端day10

文章目录前言一、web开发安全 - 攻击1. Cross-Site Scripting(XSS)1.1 存储型&#xff08;Stored Xss&#xff09;1.2 反射型&#xff08;Reflect Xss&#xff09;1.3 DOM型&#xff08;DOM Xss&#xff09;1.4 Mutation-based XSS2. Cross-Site Request Forgery&#xff08;CS…

如何使用TypeScript封装一个简单好用的Http工具

前言 Http 请求对于任何系统都是一大基石&#xff0c;那么如何封装一个高可用的 Http 请求工具呢&#xff1f;接下来手把手教你使用 TypeScript 封装一个高可用的 Http 请求工具。 本工具由三部分构成&#xff1a;Http基础层、基础方法层、业务层。 Http基础层 Http基础层主…

Lua语法入门

注意&#xff1a;文章将持续更新完善 文章目录一. 初识Lua二. HelloWorld三. Lua的数据类型四. 变量五. 循环六. 函数七. 条件控制一. 初识Lua Lua 是一种轻量小巧的脚本语言&#xff0c;用标准C语言编写并以源代码形式开放&#xff0c; 其设计目的是为了嵌入应用程序中&#…

VR全景行业的应用价值如何呈现?

互联网高速发展的今天&#xff0c;多媒体所包含的种类也是越来越多&#xff0c;而一些较为传统的表现方式已经越来越无法满足大部分客户对展示方式的要求。而在传统的表现方式中&#xff0c;展现的方式无非是静态的平面图片以及动态的视频&#xff0c;但是他们都有一个缺点就是…

MybatisPlus------BaseMapper<T>删除方法详解(三)

MybatisPlus------BaseMapper删除方法详解&#xff08;三&#xff09; MybatisPlus框架中&#xff0c;创建的mapper接口&#xff0c;需要继承BaseMapper接口,T代表与表对应的实体类。 BaseMapper接口中提供了对单表进行增删改查的基础方法。 下面分别介绍四种&#xff1a; int…

2023.2.15每日一题——867. 转置矩阵

每日一题题目描述解题核心解法一&#xff1a;二维表示 模拟解法二&#xff1a;一维表示 模拟题目描述 题目链接&#xff1a;867. 转置矩阵 给你一个二维整数数组 matrix&#xff0c; 返回 matrix 的 转置矩阵 。 矩阵的 转置 是指将矩阵的主对角线翻转&#xff0c;交换矩阵…

牛客教你用雇主品牌力抢人才!附6类校招玩法

最新校招数据显示&#xff0c;79%的应届生在Offer抉择时首要考量薪资福利。但谈钱多伤感情啊~牛客从100案例中挑出6种最潮的校招雇主品牌玩法&#xff0c;助力你抢人才。01、英特尔中国&#xff1a;“芯”动小镇雇主是否能让自己产生激情和热情&#xff0c;已经成为应届生选择O…

Sharding-jdbc读写分离

一、binlog日志查看是否开启binlog日志 show variables like log_%;注意&#xff1a;直接打开mysql-bin是乱码&#xff1b;修改my.ini配置binlog-row-imageFULLcharacter-set-serverutf8mb4default-character-setutf8mb4转成log打开首先需要切换到存放mysqlbinlog.exe应用程序的…

IDEA中使用tomcat8-maven-plugin插件

第一种方式 pom.xml <?xml version"1.0" encoding"UTF-8"?><project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://maven.apache.or…

android 混淆后的异常信息 工具处理

我们可以利用SDK中tools下的proguardgui.bat工具和混淆对应文档进行反混淆处理 D:\Android\sdk\tools\proguard\bin\proguardgui.bat 工具在SDK中的位置&#xff0c;有的SDK版本不同这个工具的具体位置可能有改变&#xff0c;也可以在tools中直接搜索proguardgui.bat&#x…

Netty 入门

文章目录一、概述1.1 Netty 是什么&#xff1f;1.2 Netty 的地位1.3 Netty 的优势二、Hello World2.1 目标2.2 服务器端2.3 客户端2.4 流程梳理三、组件3.1 EventLoop3.2 演示 NioEventLoop 处理 io 事件3.3 演示 NioEventLoop 处理普通任务3.4 演示 NioEventLoop 处理定时任务…

ThingsBoard-规则链-check existence fields

1、概述 今天我主要讲解【check existence fields】规则节点,顾名思义,这个节点就是检测字段是否存在。 可以检查规则节点消息(msg)和元数据(metadata)的字段。 2、节点理解 2.1、概述 【check existence fields】节点如图所示: 名称:你需要给这个节点命名。 Mes…

spring security简单教程以及实现完全前后端分离

简单教程spring security是spring家族的一个安全框架&#xff0c;入门简单。对比shiro&#xff0c;它自带登录页面&#xff0c;自动完成登录操作。权限过滤时支持http方法过滤。在新手入门使用时&#xff0c;只需要简单的配置&#xff0c;即可实现登录以及权限的管理&#xff0…