力扣1944.队列中可以看到的人数--单调栈

思路:

  • 由题知一个人能 看到 他右边另一个人的条件是这两人之间的所有人都比他们两人 矮 ,也就是说,在自己右边第一个比自己高的人后面的人就肯定看不到了

  • 那么只需要找到右边第一个比自己高的人与自己之间的所有满足要求的人就行了,怎么找?一个个判断中间是否有不满足要求的人吗?可行,但太慢。通过分析,不难发现在此区间内满足要求的人身高呈递增增长,也就是说,只要左边有比自己高的人那么这个人肯定看不到:

  • 那么只需要判断指定范围内每有一个满足条件的人就将能看到人数加一就行了
    代码:

    复制代码
    class Solution {
    public:
        vector<int> canSeePersonsCount(vector<int>& heights) {
            int len = heights.size();
            vector<int> answer(len);    //记录每个人可以看到几个人
    
            for(int i = 0; i < len - 1; i++){   //遍历除最后一个人外的每一个人,因为最后一个人能见人数肯定是0
                if(heights[i + 1] >= heights[i]){answer[i] = 1;continue;}   //如果右边第一个人就比自己高,直接记为1,跳过此次循环
                answer[i] = 1;   //记录当前人可见人数,因为已经判断右边第一个人,所以初值为1
                int t = heights[i + 1]; //记录左边的最高人,初值为当前人右边的人
    
                for(int j = i + 2; j < len; j++){   //从当前人右边第二个开始判断是否可见
                     if(heights[j] >= heights[i]){answer[i]++;break;}    //遇到比当前人更高的人,记录,并退出遍历
                     if(heights[j] < t) continue;    //如果左边有比自己更高的人,则跳过
                     t = heights[j];    //如果没有,则自己为当前最高的,更换最高人
                     answer[i]++;    //记录,能到这里,说明满足条件
                }
    
            }
    
            return answer;
        }
    };
  • 这个方法是我一开始的想法,没有问题,但是......

  • 真狠啊

  • 显然O(n^2)是不行了,那么就得想一个更快的方法。通过分析,不难发现在之前的方法里可以优化的点是:对于某个人,只要他左边有一个比他更高的人,那么在那个比他更高的人之前的所有人都看不到他

  • 也就是说,对于某个值,只要把能看到他的人都记录完,此时他就不被需要了可以忽视了,也就是说可以使用一个栈将每个值记录,对于不再被需要的值就弹出,在弹出的过程中刚好记录可见人数,同时将维护单调栈,从栈底到栈顶,身高严格递减

    复制代码
    假设输入为heights = [10,6,8,5,11,9],对于这个输入——

    |---|--------|------------|------------|------|---------------|
    | i | h[i] | 入栈前 | 入栈后 | 可见人数 | 解释 |
    | 5 | 9 | [] | [9] | 0 | |
    | 4 | 11 | [9] | [11] | 1 | 11挡住9,弹出9 |
    | 3 | 5 | [11] | [11,5] | 1 | |
    | 2 | 8 | [11,5] | [11,8] | 2 | 8挡住5,弹出5 |
    | 1 | 6 | [11,8] | [11,8,6] | 1 | |
    | 0 | 10 | [11,8,6] | [11,10] | 3 | 10挡住8,6,弹出8,6 |

代码:

复制代码
class Solution {
public:
    vector<int> canSeePersonsCount(vector<int>& heights) {
        int len = heights.size();
        stack<int> stack;   //栈
        vector<int> answer(len);    //记录可见人数

        for(int i = len - 1; i >= 0; i--){  //从最后一个开始压栈
            while(!stack.empty() && stack.top() < heights[i]){  //如果栈不为空且栈顶元素小于当前要压栈元素
                stack.pop();    //弹栈
                answer[i]++;    //记录
            }

            if(!stack.empty()) answer[i]++; //若栈不为空,则代表当前要压栈元素后边还有一个比之更高的人可见,记录

            stack.push(heights[i]);     //压栈
        }

        return answer;
    }
};
相关推荐
2zcode19 分钟前
基于低光照增强与轻量型CNN道路实时识别算法研究(UI界面+数据集+训练代码)
人工智能·算法·cnn·低光照增强·自动驾驶技术
郝学胜-神的一滴33 分钟前
跨平台动态库与头文件:从原理到命名的深度解析
linux·c++·程序人生·unix·cmake
代码中介商33 分钟前
C++ 仿函数(Functor)深度解析:从基础到应用
开发语言·c++
小雅痞41 分钟前
[Java][Leetcode middle] 209. 长度最小的子数组
java·算法·leetcode
王老师青少年编程43 分钟前
csp信奥赛C++高频考点专项训练之字符串 --【字符串基础】:[NOIP 2018 普及组] 标题统计
c++·字符串·csp·高频考点·信奥赛·专项训练·标题统计
做时间的朋友。1 小时前
精准核酸检测
java·数据结构·算法
空中海1 小时前
iOS 动态分析、抓包与 Frida Hook
ios·职场和发展·蓝桥杯
冯诺依曼的锦鲤1 小时前
从零实现高并发内存池:TCMalloc 核心架构拆解
c++·学习·算法·架构
Thomas_Lee_OR1 小时前
多Agent路径规划 LaCAM for multi-agent path finding (MAPF)
算法·路径规划·仓储机器人·mapf
一切皆是因缘际会1 小时前
可落地数字生命工程:从记忆厮杀到自我意识觉醒全链路,AGI内生智能硅基生命心智建模
人工智能·深度学习·算法·机器学习·ai·系统架构·agi