洛谷 P2234:[HNOI2002] 营业额统计 ← STL set

【题目来源】
https://www.luogu.com.cn/problem/P2234

【题目描述】
Tiger 最近被公司升任为营业部经理,他上任后接受公司交给的第一项任务便是统计并分析公司成立以来的营业情况。
Tiger 拿出了公司的账本,账本上记录了公司成立以来每天的营业额。分析营业情况是一项相当复杂的工作。由于节假日,大减价或者是其他情况的时候,营业额会出现一定的波动,当然一定的波动是能够接受的,但是在某些时候营业额突变得很高或是很低,这就证明公司此时的经营状况出现了问题。经济管理学上定义了一种最小波动值来衡量这种情况:当最小波动值越大时,就说明营业情况越不稳定。
而分析整个公司的从成立到现在营业情况是否稳定,只需要把每一天的最小波动值加起来就可以了。你的任务就是编写一个程序帮助 Tiger 来计算这一个值。
我们定义,一天的最小波动值 = min{∣该天以前某一天的营业额−该天营业额∣}。
特别地,第一天的最小波动值为第一天的营业额。

【输入格式】
第一行为正整数 n(n≤32767) ,表示该公司从成立一直到现在的天数,接下来的 n 行每行有一个整数 ai(∣ai∣≤10^6) ,表示第 i 天公司的营业额,可能存在负数。

【输出格式】
输出一个正整数,即每一天最小波动值的和,保证结果小于 2^31。

【输入样例】
6
5
1
2
5
4
6

【输出样例】
12

【说明/提示】
结果说明:5+∣1−5∣+∣2−1∣+∣5−5∣+∣4−5∣+∣6−5∣=5+4+1+0+1+1=12

【算法分析】
● STL set 常用函数解析
https://blog.csdn.net/hnjzsyjyj/article/details/127017796
https://blog.csdn.net/hnjzsyjyj/article/details/145528031

● 代码逻辑解析
‌(一)初始化阶段‌
在集合 s 中预先插入 inf 和 -inf,确保后续查找操作始终存在前驱和后继节点,避免空集合导致的异常‌。**注意:**此处的无穷大 inf 设为 0x3f3f3f3f,不要设为 0x7f7f7f7f。这是因为,0x7f7f7f7f 的缺点是容易在加法运算中溢出,导致负数结果,这在算法中可能引发错误‌。
‌(二)输入处理阶段‌
读取整数 n,循环处理每个输入值 x。
‌1.当集合仅含初始边界 inf 和 -inf 时‌(s.size() == 2),直接插入 x 并将 x 累加到答案 ans 中。
‌2. 当集合已有其他元素时‌:
(1)使用 lower_bound(x) 找到第一个大于等于 x 的迭代器 it‌。
(2)若 *it != x(即 x 不存在于集合中),计算 x 与 *it(后继)和 *--t(前驱)的最小差值,累加到 ans。
(3)插入 x 到集合中。
(三)输出结果‌
最终输出累加的最小差值总和 ans。

● 计算过程详析
1.输入 5 时 → 集合初始只有两个边界 → ans+=5 → 插入 5
2.输入 1 时 → 前驱 -inf,后继 5 → 最小差值 4 → ans+=4 → 插入 1
3.输入 2 时 → 前驱 1,后继 5 → 最小差值1 → ans+=1 → 插入 2
4.输入 5 时 → 已存在,不处理
5.输入 4 时 → 前驱 2,后继 5 → 最小差值 1 → ans+=1 → 插入 4
6.输入 6 时 → 前驱 5,后继 inf → 最小差值 1 → ans+=1 → 插入 6
累计总和:5+4+1+1+1=12

● 适用场景
该算法适用于需要动态维护有序序列,并在每次插入时快速计算与相邻元素的最小差值的场景,如实时数据流分析或特定竞赛题目‌。

【算法代码】

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

const int inf=0x3f3f3f3f;
set<int> s;
set<int>::iterator it,t;
int x,ans;
int n;

int main() {
    s.insert(inf);
    s.insert(-inf);

    cin>>n;
    while(n--) {
        cin>>x;
        if(s.size()==2) {
            s.insert(x);
            ans+=x;
        } else {
            it=s.lower_bound(x);
            if(*it!=x) {
                t=it, t--;
                ans+=min(abs(x-*it),abs(x-*t));
                s.insert(x);
            }
        }
    }
    cout<<ans<<endl;

    return 0;
}

/*
in:
6
5
1
2
5
4
6

out:
12
*/

【参考文献】
https://blog.csdn.net/hnjzsyjyj/article/details/145528031
https://blog.csdn.net/hnjzsyjyj/article/details/146110033

相关推荐
Jay_51510 天前
C++ STL 模板详解:由浅入深掌握标准模板库
c++·学习·stl
让我们一起加油好吗12 天前
【基础算法】贪心 (一) :简单贪心
c++·算法·贪心算法·stl·洛谷·牛客
DARLING Zero two♡17 天前
C++数据的输入输出秘境:IO流
c++·stl·io流
看到我,请让我去学习19 天前
C++核心编程(动态类型转换,STL,Lanmda)
c语言·开发语言·c++·stl
君鼎20 天前
C++标准库大全(STL)
开发语言·c++·stl
秦少游在淮海25 天前
C++ - string 的使用 #auto #范围for #访问及遍历操作 #容量操作 #修改操作 #其他操作 #非成员函数
开发语言·c++·stl·string·范围for·auto·string 的使用
Dovis(誓平步青云)1 个月前
探索C++标准模板库(STL):String接口的底层实现(下篇)
开发语言·c++·stl·string
SunkingYang1 个月前
C++中如何遍历map?
c++·stl·map·遍历·方法
YKPG1 个月前
C++学习-入门到精通【13】标准库的容器和迭代器
c++·学习·stl
EutoCool1 个月前
【项目】在线OJ(负载均衡式)
运维·c++·stl·负载均衡