发射站

题目描述

某地有 \(N\) 个能量发射站排成一行,每个发射站 \(i\) 都有不相同的高度 \(H_i\),并能向两边(两端的发射站只能向一边)同时发射能量值为 \(V_i\) 的能量,发出的能量只被两边最近的且比它高的发射站接收。显然,每个发射站发来的能量有可能被 \(0\)\(1\)\(2\) 个其他发射站所接受。

请计算出接收最多能量的发射站接收的能量是多少。

输入格式

\(1\) 行一个整数 \(N\)

\(2\)\(N+1\) 行,第 \(i+1\) 行有两个整数 \(H_i\)\(V_i\),表示第 \(i\) 个人发射站的高度和发射的能量值。

输出格式

输出仅一行,表示接收最多能量的发射站接收到的能量值。答案不超过 32 位带符号整数的表示范围。

样例 #1

样例输入 #1

3
4 2 
3 5 
6 10

样例输出 #1

7

提示

对于 \(40\%\) 的数据,\(1\le N\le 5000,1\le H_i\le 10^5,1\le V_i\le 10^4\)

对于 \(70\%\) 的数据,\(1\le N\le 10^5,1\le H_i\le 2\times 10^9,1\le V_i\le 10^4\)

对于 \(100\%\) 的数据,\(1\le N\le 10^6,1\le H_i\le 2\times 10^9,1\le V_i\le 10^4\)

分析

  • 法1:直接暴力模拟,左右查找合适的值,可以过 40%,会 TLE。
  • 法2:维护单调递增栈(栈顶到栈底元素单调递增)

image

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=1e6+10,INF=0x3f3f3f3f;
int n,h[N],v[N],sta[N],head=0;
LL ans[N];

void slove1() { // 预计 TLE
    for(int i=1; i<=n; i++) {
        int l=i-1, r=i+1;
        while(l>0 && h[l]<=h[i]) l--;
        while(r<=n && h[r]<=h[i]) r++;
        ans[l]+=v[i], ans[r]+=v[i];
    }
}
void slove2() { // 维护一个单调递增栈
    for(int i=1; i<=n; i++) {
        while(head && h[sta[head]] < h[i]) {
            ans[i] += v[sta[head]], head--;
        }
        ans[sta[head]] += v[i];
        sta[++head] = i;
    }
}
int main() {
//    freopen("data.in", "r", stdin);
    scanf("%d",&n);
    for(int i=1; i<=n; i++) scanf("%d%d",&h[i],&v[i]);
    slove2();
    for(int i=1; i<=n; i++) ans[1]=max(ans[1],ans[i]);
    printf("%lld\n",ans[1]);
    return 0;
}

原文地址:http://www.cnblogs.com/hellohebin/p/16800242.html

1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长! 2. 分享目的仅供大家学习和交流,请务用于商业用途! 3. 如果你也有好源码或者教程,可以到用户中心发布,分享有积分奖励和额外收入! 4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解! 5. 如有链接无法下载、失效或广告,请联系管理员处理! 6. 本站资源售价只是赞助,收取费用仅维持本站的日常运营所需! 7. 如遇到加密压缩包,默认解压密码为"gltf",如遇到无法解压的请联系管理员! 8. 因为资源和程序源码均为可复制品,所以不支持任何理由的退款兑现,请斟酌后支付下载 声明:如果标题没有注明"已测试"或者"测试可用"等字样的资源源码均未经过站长测试.特别注意没有标注的源码不保证任何可用性