【Leetcode】881. 救生艇

文章目录

题目

点击打开题目链接🔗

给定数组 p e o p l e people people 。 p e o p l e i peoplei peoplei表示第 i i i 个人的体重 ,船的数量不限 ,每艘船可以承载的最大重量为 l i m i t limit limit。

每艘船最多可同时载两人 ,但条件是这些人的重量之和最多为 l i m i t limit limit。

返回 承载所有人所需的最小船数 。

示例 1
输入 :people = 1,2, limit = 3
输出 :1
解释:1 艘船载 (1, 2)
示例 2
输入 :people = 3,2,2,1, limit = 3
输出 :3
解释:3 艘船分别载 (1, 2), (2) 和 (3)
示例 3
输入 :people = 3,5,3,4, limit = 5
输出 :4
解释:4 艘船分别载 (3), (3), (4), (5)
提示
1 ≤ p e o p l e . l e n g t h ≤ 5 ∗ 1 0 4 1 \leq people.length \leq 5 * 10^4 1≤people.length≤5∗104
1 ≤ p e o p l e i ≤ l i m i t ≤ 3 ∗ 1 0 4 1 \leq peoplei \leq limit \leq 3 * 10^4 1≤peoplei≤limit≤3∗104

思路

为了最小化船只的数量,可以使用贪心的策略,可以想到将最轻的人和最重的人安排在一起,如果他们的重量之和小于或者是等于 l i m i t limit limit,那么他们就是可以一起坐在一艘船,然后我们在剩下的人中继续找最轻的和最重的继续验证;如果最轻的和最重的不能坐一艘船,那么最重的人和剩下的任意一个人都无法配对坐同一艘船,最重的那个人需要独自坐一艘船;以上反复操作知道全部人分配完毕。

总结一下步骤:

  1. 首先,将数组 people 进行排序,这样可以方便地从最轻和最重的人开始配对
  2. 使用两个指针:一个指向最轻的人的起始位置(左指针),一个指向最重的人的末尾位置(右指针)
  3. 尝试将最轻和最重的人配对。如果他们的体重之和小于或等于 limit,他们可以一起坐一艘船,此时左指针右移,右指针左移。
  4. 如果最轻和最重的人不能一起坐一艘船(即他们的体重之和大于 limit),那么最重的人独自坐一艘船,此时右指针左移
  5. 每次操作,船的数量加一
  6. 步骤 3 3 3、 4 4 4、 5 5 5重复执行直到所有的人都安排到船上

代码

c++ 复制代码
class Solution {
public:
    int numRescueBoats(vector<int>& people, int limit) {
        sort(people.begin(),people.end());
        int minboat=0;
        for(int left=0,right=people.size()-1;left<=right;right--)
        {
            if(people[left]+people[right]<=limit)left++;
            minboat++;
        }
        return minboat;
    }
};

复杂度分析

时间复杂度

  • 排序的时间复杂度为 O ( n log ⁡ n ) O(n \log n) O(nlogn),其中这个 n n n是数组 p e o p l e people people的长度
  • 双指针遍历数组的时间复杂度是 O ( n log ⁡ n ) O(n \log n) O(nlogn)
    因此总的时间复杂度是 O ( n log ⁡ n ) O(n \log n) O(nlogn)

空间复杂度

  • 排序的空间复杂度取决于排序的算法。在C++中, s o r t sort sort函数通常使用的是 i n t r o s o r t introsort introsort,其空间复杂度为 O ( log ⁡ n ) O(\log n) O(logn)
  • 其余部分使用的空间为常数级别为 O ( 1 ) O(1) O(1)
    总的来说空间复杂度为 O ( log ⁡ n ) O(\log n) O(logn)

结果



ps:这里不太清楚为啥空间复杂度能干到常数级别,可能是因为数组长度太小了只有 5 ∗ 1 0 4 5 * 10^4 5∗104?有没有大佬说说

总结

该问题使用贪心算法和双指针技巧,通过先排序,再使用双指针从最轻和最重的人开始配对,使得每次尽可能转俩人,从而最小化所需要的船的数量。

相关推荐
彷徨而立5 小时前
【C++】介绍 std::ifstream 输入文件流
开发语言·c++
MC皮蛋侠客5 小时前
C++17 多线程系列(十):多线程性能优化——从测量到调优
c++·多线程
程序大视界6 小时前
【C++ 从基础到项目实战】C++(六):拷贝控制——浅拷贝与深拷贝,兼谈智能指针
开发语言·c++·cpp
妄想出头的工业炼药师6 小时前
LVIO鲁棒
算法·开源
aini_lovee6 小时前
MATLAB 图像修复 — 偏微分方程方法
算法
Purple Coder6 小时前
MgB2论文草稿1
职场和发展
Cthy_hy6 小时前
Python算法竞赛:排列组合核心用法
开发语言·python·算法
大圣编程6 小时前
面向对象深度理解
java·开发语言·算法
爱喝水的鱼丶6 小时前
SAP-ABAP:SAP 简单报表输出开发系列(共6篇) 第四篇:SAP 报表异常处理机制:数据校验与消息提示规范落地
开发语言·数据库·学习·算法·sap·abap
代码中介商7 小时前
C++四大设计模式:单例、工厂、观察者、策略
java·c++·设计模式