洛谷 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

相关推荐
BigBookX3 天前
从一个问题回忆C++中std::map的初始化过程
c++·stl·map
阿猿收手吧!4 天前
【CPP】STL容器模拟实现篇之string
开发语言·c++·容器·stl
笨手笨脚の9 天前
Redis 源码分析-内部数据结构 intset
数据结构·redis·set·intset·并差集
努力学习的小廉11 天前
【AVL树】—— 我与C++的不解之缘(二十三)
开发语言·c++·stl
星迹日15 天前
数据结构:Map & set - 习题(三)
java·数据结构·经验分享·set·map
小小小白的编程日记18 天前
List的基本功能(1)
数据结构·c++·算法·stl·list
DARLING Zero two♡24 天前
C++效率掌握之STL库:string底层剖析
开发语言·c++·stl·string
laimaxgg25 天前
C++ STL容器之list的使用及复现
c++·windows·容器·stl·list
surtr11 个月前
【C++】RBTree(红黑树)模拟实现
数据结构·c++·算法·stl·map·红黑树·rbtree