Leetcode452. 用最少数量的箭引爆气球

news/2024/11/25 7:44:31/

Every day a Leetcode

题目来源:452. 用最少数量的箭引爆气球

解法1:排序 + 贪心

题解:用最少数量的箭引爆气球

我们首先随机地射出一支箭,再看一看是否能够调整这支箭地射出位置,使得我们可以引爆更多数目的气球。

在这里插入图片描述

如图 1-1 所示,我们随机射出一支箭,引爆了除红色气球以外的所有气球。我们称所有引爆的气球为「原本引爆的气球」,其余的气球为「原本完好的气球」。可以发现,如果我们将这支箭的射出位置稍微往右移动一点,那么我们就有机会引爆红色气球,如图 1-2 所示。

那么我们最远可以将这支箭往右移动多远呢?我们唯一的要求就是:原本引爆的气球只要仍然被引爆就行了。这样一来,我们找出原本引爆的气球中右边界位置最靠左的那一个,将这支箭的射出位置移动到这个右边界位置,这也是最远可以往右移动到的位置:如图 1-3 所示,只要我们再往右移动一点点,这个气球就无法被引爆了。

为什么「原本引爆的气球仍然被引爆」是唯一的要求?别急,往下看就能看到其精妙所在。

因此,我们可以断定:

一定存在一种最优(射出的箭数最小)的方法,使得每一支箭的射出位置都恰好对应着某一个气球的右边界。

这是为什么?我们考虑任意一种最优的方法,对于其中的任意一支箭,我们都通过上面描述的方法,将这支箭的位置移动到它对应的「原本引爆的气球中最靠左的右边界位置」,那么这些原本引爆的气球仍然被引爆。这样一来,所有的气球仍然都会被引爆,并且每一支箭的射出位置都恰好位于某一个气球的右边界了。

有了这样一个有用的断定,我们就可以快速得到一种最优的方法了。考虑所有气球中右边界位置最靠左的那一个,那么一定有一支箭的射出位置就是它的右边界(否则就没有箭可以将其引爆了)。当我们确定了一支箭之后,我们就可以将这支箭引爆的所有气球移除,并从剩下未被引爆的气球中,再选择右边界位置最靠左的那一个,确定下一支箭,直到所有的气球都被引爆。

我们可以写出如下的伪代码:

let points := [[x(0), y(0)], [x(1), y(1)], ... [x(n-1), y(n-1)]],表示 n 个气球
let burst := [false] * n,表示每个气球是否被引爆
let ans := 1,表示射出的箭数将 points 按照 y 值(右边界)进行升序排序while burst 中还有 false 值 dolet i := 最小的满足 burst[i] = false 的索引 ifor j := i to n-1 doif x(j) <= y(i) thenburst[j] := trueend ifans := ans + 1end for
end whilereturn ans

这样的做法在最坏情况下时间复杂度是 O(n2),即这 n 个气球对应的区间互不重叠,while 循环需要执行 n 次。

代码:

/** @lc app=leetcode.cn id=452 lang=cpp** [452] 用最少数量的箭引爆气球*/// @lc code=start
class Solution
{
private:static bool cmp(const vector<int> A, const vector<int> B){return A[1] < B[1];}bool remainBalloons(vector<bool> &burst){for (int i = 0; i < burst.size(); i++)if (burst[i] == false)return true;return false;}public:int findMinArrowShots(vector<vector<int>> &points){if (points.empty())return 0;int n = points.size();vector<bool> burst(n, false);int ans = 0;sort(points.begin(), points.end(), cmp);while (remainBalloons(burst)){int i = 0;while (i < n && burst[i] == true)i++;for (int j = i; j < n; j++){if (points[j][0] <= points[i][1])burst[j] = true;}ans++;}return ans;}
};
// @lc code=end

结果:超时

在这里插入图片描述

复杂度分析:

时间复杂度:O(n2),其中 n 是数组 points 的长度。排序的时间复杂度为 O(nlogn),对所有气球进行遍历并计算答案的时间复杂度为 O(n),其在渐进意义下小于前者,因此可以忽略。

空间复杂度:O(n),其中 n 是数组 points 的长度。我们用到了长度为 n 的复制数组 burst。

解法2:

那么我们如何继续进行优化呢?

事实上,在内层的 j 循环中,当我们遇到第一个不满足 x(j)≤y(i) 的 j 值,就可以直接跳出循环,并且这个 y(j) 就是下一支箭的射出位置。为什么这样做是对的呢?我们考虑某一支箭的索引 it 以及它的下一支箭的索引 jt,对于索引在 jt 之后的任意一个可以被 it 引爆的气球,记索引为 j0 ,有:x(j0)≤y(it)。

由于 y(it)≤y(jt) 显然成立,那么 x(j0)≤y(jt) 也成立,也就是说:当前这支箭在索引 jt(第一个无法引爆的气球)之后所有可以引爆的气球,下一支箭也都可以引爆。因此我们就证明了其正确性,也就可以写出如下的伪代码:

let points := [[x(0), y(0)], [x(1), y(1)], ... [x(n-1), y(n-1)]],表示 n 个气球
let pos := y(0),表示当前箭的射出位置
let ans := 1,表示射出的箭数将 points 按照 y 值(右边界)进行升序排序for i := 1 to n-1 doif x(i) > pos thenans := ans + 1pos := y(i)end if
end forreturn ans

这样就可以将计算答案的时间从 O(n2) 降低至 O(n)。

代码:

class Solution
{
private:static bool cmp(const vector<int> &A, const vector<int> &B){return A[1] < B[1];}public:int findMinArrowShots(vector<vector<int>> &points){if (points.empty())return 0;// sort(points.begin(), points.end(), [](const vector<int> &u, const vector<int> &v)//      { return u[1] < v[1]; });sort(points.begin(), points.end(), cmp);int pos = points[0][1];int ans = 1;for (const vector<int> &balloon : points){if (balloon[0] > pos){pos = balloon[1];ans++;}}return ans;}
};

结果:

在这里插入图片描述

复杂度分析:

时间复杂度:O(nlogn),其中 n 是数组 points 的长度。排序的时间复杂度为 O(nlogn),对所有气球进行遍历并计算答案的时间复杂度为 O(n),其在渐进意义下小于前者,因此可以忽略。

空间复杂度:O(logn),其中 n 是数组 points 的长度。即为排序需要使用的栈空间。


http://www.ppmy.cn/news/75146.html

相关文章

Godot引擎 4.0 文档 - 入门介绍 - Godot简介

本文为Google Translate英译中结果&#xff0c;DrGraph在此基础上加了一些校正。英文原版页面&#xff1a;Introduction to Godsot — Godot Engine (stable) documentation in English Godot简介 本文旨在帮助您确定 Godot 是否适合您。我们将介绍该引擎的一些广泛功能&#…

CMD与DOS脚本编程【第七章】

预计更新 第一章. 简介和基础命令 1.1 介绍cmd/dos脚本语言的概念和基本语法 1.2 讲解常用的基础命令和参数&#xff0c;如echo、dir、cd等 第二章. 变量和运算符 2.1 讲解变量和常量的定义和使用方法 2.2 介绍不同类型的运算符和运算规则 第三章. 控制流程和条件语句 3.1 介…

运行100万个并发任务,不同语言各需要多少内存

作者&#xff1a;DataStax 公司&#xff08;美国的一家数据库系统开发商&#xff09;Piotr Kołaczkowski 原文见&#xff1a; https://pkolaczk.github.io/memory-consumption-of-async/ 在这篇博客文章中&#xff0c;探讨了处理大量网络连接时候的Rust、Go、Java、C#、Pyth…

06 常用类2

StringBuiulder的常用方法和作用 作用:为字符串数组提供了增删改查的方法 注意: StringBuffer():默认容量为16的可变字符串 StringBuffer( int capacity):初始化容量就是里面字符串的长度 常用的方法:append inset delete setCharAt replace StringBuffer sbnew StringBu…

【C语言】入门必看之循环练习(含二分查找动图)

&#x1f6a9;纸上得来终觉浅&#xff0c; 绝知此事要躬行。 &#x1f31f;主页&#xff1a;June-Frost &#x1f680;专栏&#xff1a;C语言 ⚡注&#xff1a;此篇文章的 代码风格部分 将根据《高质量 C/C 编程指南》 —— 林锐 进行说明。该部分将用紫色表示 该篇将对循环语…

【bsauce读论文】2023-SP-内核Use-After-Cleanup漏洞挖掘与利用

本文参考G.O.S.S.I.P 阅读推荐 2023-01-06 UACatcher做一些补充。 1. UAC漏洞介绍 UAC漏洞介绍&#xff1a;Use-After-Cleanup &#xff08;UAC&#xff09;漏洞类似UAF&#xff0c;本文主要检测Linux内核中UAC漏洞。UAC基本原理参见图Fig-1。首先&#xff0c;UAC漏洞和系统中…

Java 实现实时监听MySQL数据库变更MySQLBinListener

目录 1、导出需要的类和接口 2、 定义 MySQLBinlogListener类 3、私有方法&#xff0c;启动重连定时器 4、完整代码 编写一个MySQL数据库实时变更的监听器。 为什么要编写这个一个监听器&#xff1a;为了实时监测和响应MySQL数据库中的变更事件 实时数据同步&#xff1a;通…

springboot实现webocket长连接(四)--netty+websocekt+服务端

springboot实现webocket长连接&#xff08;四&#xff09; demo下载地址&#xff1a;多种websocket实现方式&#xff0c;其中有基于spring-websocekt&#xff0c;也有基于netty框架&#xff0c;即下即用。 之前的博客使用了spring-websocket实现了websocket服务端&#xff0c…