【题解】Diferenc-Diferencija [SP10622]

传送门:\(\text{Diferenc-Diferencija}\) \(\text{[SP10622]}\)

【题目描述】

序列的值被定义成其中最大的元素减去最小的元素。如序列 \((3,1,7,2)\) 的值为 \(7-1=6\), 序列 \((42,42)\) 的值为 \(42-42=0\)。

现给定一长为 \(n\) 的序列 \(a\),求出所有连续子序列的值的和。

【样例】

样例输入:
3
1
2
3
样例输出:
4 样例输入:
4
7
5
7
5
样例输出:
12 样例输入:
4
3
1
7
2
样例输出:
31

【数据范围】

\(100 \%:\) \(2 \leqslant n \leqslant 3*10^5,\) \(1 \leqslant a[i] \leqslant 10^8\)


【分析】

先将子区间的右端点固定为 \(r\),此时一共有 \(r\) 个左端点 \((l \in [1,r])\) 可与之组成连续子序列,用 \(f_1[l]\) 表示 \(max \{a[j]\}(j \in [l,r])\) ,\(f_2[l]\) 表示 \(min \{a[j]\}(j \in [l,r])\) 。于是以 \(i\) 为右端点 \(r\) 的子区间贡献和为 \(\sum_{l=1}^{r} (f_1[l]-f_2[l])\),即 \(\sum_{l=1}^{r} f_1[l] - \sum_{l=1}^{r} f_2[l]\) 。我们可以分开算 \(f_1,f_2\) 的总和。

当右端点移至 \(r+1\) 时,需要用 \(a[r+1]\) 来更新 \(f_1,f_2\),可以直接扫描 \([1,r]\),但时间不过不了关。

随着 \(l\) 的减小,\(f_1[l]=max(a[l],f_1[l+1])\),可以发现 \(f_1[l]\) 是单调不下降的,同理 \(f_2[l]\) 单调不上升。

随着 \(r\) 的增大,\(f_1[l]=max(f_1[l],a[r+1])\),可以发现 \(f_1[l]\) 仍是单调不下降的,同理 \(f_2[l]\) 单调不上升。

然后我们就会发现一个现象:每次新加进来一个数 \(a[r+1]\) 时,它会将以 \(r+1\) 结尾的一段连续的区间 \(f_1[l],f_2[l](l \in [?,r+1])\) 全部赋值为 \(a[r+1]\),而且被覆盖掉的原数对这之后的区间不再有任何贡献。

于是我们可以用两个单调栈分别维护 \(f_1,f_2\),由于下标也是单调递增,所以可以将 \(f\) 值相同的合并起来,用 \(g\) 记录 \(f\) 相同的下标个数,另设一个变量 \(S\) 表示以 \(i\) 为右端点的贡献和,当加入新的 \(a[r]\) 时就不断弹走队尾直至保持单调时结束,同时更新 \(S\),最后累加答案即可。

【Code】

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#define LL long long
#define Re register int
using namespace std;
const int N=3e5+2;
int n,t1,t2,a[N],f1[N],f2[N],g1[N],g2[N];LL S,ans;
inline void in(Re &x){
int f=0;x=0;char c=getchar();
while(c<'0'||c>'9')f|=c=='-',c=getchar();
while(c>='0'&&c<='9')x=(x<<1)+(x<<3)+(c^48),c=getchar();
x=f?-x:x;
}
int main(){
in(n);
for(Re i=1;i<=n;++i)in(a[i]);
f1[++t1]=a[1],f2[++t2]=a[1],g1[t1]=g2[t2]=1,ans=S=0;//初始化入队
for(Re i=2;i<=n;++i){
Re tmp=1;//f1[i]和f2[i]都一定会被覆盖,所以初始化为1
while(t1&&f1[t1]<=a[i])S-=(LL)f1[t1]*g1[t1],tmp+=g1[t1--];//更新最大值
f1[++t1]=a[i],g1[t1]=tmp,S+=(LL)a[i]*tmp;
tmp=1;
while(t2&&f2[t2]>=a[i])S+=(LL)f2[t2]*g2[t2],tmp+=g2[t2--];//更新最小值
f2[++t2]=a[i],g2[t2]=tmp,S-=(LL)a[i]*tmp;
ans+=S;//累加答案
}
printf("%lld\n",ans);
return 0;
}

【题解】Diferenc-Diferencija [SP10622]的更多相关文章

  1. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  2. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  3. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  4. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

  5. 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解

    题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...

  6. 2016ACM青岛区域赛题解

    A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Jav ...

  7. poj1399 hoj1037 Direct Visibility 题解 (宽搜)

    http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...

  8. 网络流n题 题解

    学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...

  9. CF100965C题解..

    求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...

随机推荐

  1. Hive的数据倾斜

    目录 什么是数据倾斜 Hadoop框架的特性 主要表现 容易数据倾斜的情况 产生数据清洗的原因 业务场景 空值产生的数据倾斜 不同数据类型关联产生数据倾斜 大小表关联查询产生数据倾斜 一.什么是数据倾 ...

  2. TP5 Request 请求对象【转】

    app\index\controller\Index.php <?php namespace app\index\controller; use think\Request; class Ind ...

  3. Django forms 主要的标签介绍

    修改 forms.py from django import forms as DForms from django.forms import fields from django.forms imp ...

  4. iptables 学习笔记

    1. 安装iptables yum install iptables centos7: yum install -y iptables-services 2. service命令 查看iptables ...

  5. Linux下CPU利用率和负载的关系

    1.CPU利用率和负载 CPU利用率显示的是程序在运行期间实时占用的CPU百分比:cpu使用率反映的是当前cpu的繁忙程度,忽高忽低的原因在于占用cpu处理时间的进程可能处于io等待状态但却还未释放进 ...

  6. 2015年第六届蓝桥杯C/C++程序设计本科B组决赛 ——居民集会(编程大题)

    标题:居民集会 蓝桥村的居民都生活在一条公路的边上,公路的长度为L,每户家庭的 位置都用这户家庭到公路的起点的距离来计算,第i户家庭距起点的距 离为di.每年,蓝桥村都要举行一次集会.今年,由于村里的 ...

  7. python调用oracle存储过程

    oracle 存储过程 python调用oracle存储过程 -- 通过cx_Oracle连接 import cx_Oracle # 连接数据库 orcl_engine = 'scott/s123@x ...

  8. 14-numpy笔记-莫烦pandas-2

    代码 import pandas as pd import numpy as np dates = pd.date_range('20130101', periods=6) df=pd.DataFra ...

  9. USACO Ski Course Design

    洛谷P3650 https://www.luogu.org/problemnew/show/P3650 JDOJ 2393 https://neooj.com:8082/oldoj/problem.p ...

  10. 使用element-ui的table组件时,渲染为html格式

    背景 今天在做vue的项目时,使用到 element-ui 的 table 组件,使用富文本编辑器进行新增操作后,发现 html格式 并没有被识别 原因 在 element-ui 中,table组件默 ...