题面

传送门

题目大意:

L(h)的值是区间[L,R]内,abs(h[i]-h[j])/(i-j)的最大值。现在有q个询问,每个询问表示询问区间[L,R]内,所有子序列的L(h)的值的和

分析

将|h[i]−h[j]i−j|" role="presentation" style="position: relative;">∣∣∣h[i]−h[j]i−j∣∣∣|h[i]−h[j]i−j|想成斜率,显然选相邻的两个数最优,最大的斜率只会存在于相邻两点

所以我们可以预处理所有h[i]-h[i-1]的值,记作d[i]

问题转化为求[L,R]中每个子区间中的最大d[i]之和

朴素算法是枚举所有子区间,时间复杂度O(n2)" role="presentation" style="position: relative;">O(n2)O(n2),显然会TLE

因此我们可以计算每个d[i]被算了多少次

具体方法如下:

维护一个单调栈,从左到右依次将d[i]入栈,栈顶元素最小

那么过程中,当区间左端点<=i-1时,区间最大值为弹出栈顶的元素

当区间右端点>=i+1时,区间最大值为新加入进去的元素

所谓x的作用,就是区间端点在a左侧或在b右侧时,区间内的最大值为x,此时便是”有作用”的,因为会被算进结果里

如d={2,5,3,9,8,1}

将5入栈时弹出2,则区间右端点小于等于1时,区间最大值为2

当区间左端点>=2时,区间最大值为5 (当然此时还没有计算出右端点,之后9入栈可以计算出右端点<=3时区间最大值为5)

设lbound[i],rbound[i]表示区间左端点>=lbound[i]且区间右端点<=rbound[i]时区间最大值为d[i],查询区间为[L,R]

则由乘法原理得,d[i]被计算了(i-lbound[i]+1)*(rbound[i]-i+1)次

于是对于每一个i,我们可以O(1)" role="presentation" style="position: relative;">O(1)O(1)的时间内求出结果

单调栈预处理时间复杂度O(n)" role="presentation" style="position: relative;">O(n)O(n),查询时间复杂度为O(nq)" role="presentation" style="position: relative;">O(nq)O(nq)

一些陷阱:

1.数据范围:最后答案要用long long

2.单调栈最后里面元素要出栈,它们的rbound值为n

3.若单调栈为空,新插入的元素的lbound值为1

代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<stack>
#include<algorithm>
#include<cmath>
#define maxn 100005
#define ForMyLove return 0;
using namespace std;
int n,q;
int a[maxn];
int lbound[maxn],rbound[maxn];
long long d[maxn];
struct node{
int pos;
int value;
node(){ }
node(int i,int x){
pos=i;
value=x;
}
};
stack<node>s;
int main(){
int l,r;
scanf("%d %d",&n,&q);
for(int i=1;i<=n;i++) scanf("%d",&a[i]);
for(int i=1;i<=n;i++) d[i]=abs(a[i]-a[i-1]);
for(int i=1;i<=n;i++){
node now;
while(!s.empty()){
now=s.top();
if(d[i]>now.value){
rbound[now.pos]=i-1;
s.pop();
}else break;
}
if(s.size()==0) lbound[i]=1;
else{
now=s.top();
lbound[i]=now.pos+1;
}
s.push(node(i,d[i]));
}
while(!s.empty()){
node now=s.top();
rbound[now.pos]=n;
s.pop();
}
while(q--){
scanf("%d %d",&l,&r);
long long ans=0;
for(int i=l+1;i<=r;i++){
long long left=max(l+1,lbound[i]);
long long right=min(r,rbound[i]);
ans=ans+(i-left+1)*(right-i+1)*d[i];
}
printf("%I64d\n",ans);
}
ForMyLove
}

Codeforces 601B(贪心+斜率+组合数学+单调栈)的更多相关文章

  1. Codeforces 601B. Lipshitz Sequence(单调栈)

    Codeforces 601B. Lipshitz Sequence 题意:,q个询问,每次询问给出l,r,求a数组[l,r]中所有子区间的L值的和. 思路:首先要观察到,斜率最大值只会出现在相邻两点 ...

  2. 【BZOJ1007】【HNOI2008】水平可见直线(斜率排序+单调栈)

    1007: [HNOI2008]水平可见直线 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 2605  Solved: 914[Submit][Stat ...

  3. 【BZOJ4709】[Jsoi2011]柠檬 斜率优化+单调栈

    [BZOJ4709][Jsoi2011]柠檬 Description Flute 很喜欢柠檬.它准备了一串用树枝串起来的贝壳,打算用一种魔法把贝壳变成柠檬.贝壳一共有 N (1 ≤ N ≤ 100,0 ...

  4. codeforces 547B. Mike and Feet 单调栈

    题目链接 用单调栈计算出一个数字, 左边第一个比他小的数字的位置, 右边比第一个他小的数字的位置, 然后len = r[i] - l[i] +1. ans[len] = max(ans[len], a ...

  5. Codeforces 547B. Mike and Feet[单调栈/队列]

    这道题用单调递增的单调栈维护每个数能够覆盖的最大区间即可. 对于   1 2 3 4 5 4 3 2 1 6 这组样例, 1能够覆盖的最大区间是10,2能够覆盖的最大区间是7,以此类推,我们可以使用单 ...

  6. CF535E Tavas and Pashmaks 单调栈、凸包

    传送门 题意:有一场比赛,$N$个人参加.每个人有两种参数$a,b$,如果存在正实数$A,B$使得$\frac{A}{a_i} + \frac{B}{b_i}$在$i=x$处取得最大值(可以有多个最大 ...

  7. CodeForces 548D 单调栈

    Mike and Feet Time Limit:1000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64u Subm ...

  8. [CF442C] Artem and Array (贪心+单调栈优化)

    题目链接:http://codeforces.com/problemset/problem/442/C 题目大意:一个数列,有n个元素.你可以做n-2次操作,每次操作去除一个数字,并且得到这个数字两边 ...

  9. Codeforces Round #333 (Div. 1)--B. Lipshitz Sequence 单调栈

    题意:n个点, 坐标已知,其中横坐标为为1~n. 求区间[l, r] 的所有子区间内斜率最大值的和. 首先要知道,[l, r]区间内最大的斜率必然是相邻的两个点构成的. 然后问题就变成了求区间[l, ...

随机推荐

  1. bzoj4025 二分图 LCT + 最小生成树

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4025 题解 貌似这道题有一个非常简单的做法是线段树分治+并查集. 可是我是为了练 LCT 来做 ...

  2. ARM指令集的最新版本包括针对JavaScript的优化

    在ARM指令集中,ARMv8.3添加了一个新的float-to-int指令,其错误和超出范围的值按照JavaScript的方式处理.以前[指令]获取JavaScript的语义要慢得多,JavaScri ...

  3. BZOJ1896 Equations 线性规划+半平面交+三分

    题意简述 给你\(3\)个数组\(a_i\),\(b_i\)和\(c_i\),让你维护一个数组\(x_i\),共\(m\)组询问,每次给定两个数\(s\),\(t\),使得 \[ \sum_i a_i ...

  4. Java内存模型(JMM)的可见性

    JMM(Java Memory Model)内存模型之可见性 JMM是Java内存模型的缩写,本身是一种抽象的概念,并不真实存在,它描述的是一组规则或规范,通过这组规范定义了程序中各个变量(包括实例字 ...

  5. hadoop中yarn

    一.yarn的概述 Apache Yarn(Yet Another Resource Negotiator的缩写)是hadoop集群资源管理器系统,Yarn从hadoop 2引入,最初是为了改善Map ...

  6. APP功能测试注意点

    App功能测试的7大注意点 : APP测试   在日常工作的摸索中,我们将如何做好app测试的注意点简单归结为如下内容.  弱网测试,兼容性测试,UI测试.中断测试, 01 运行 1)App安装完成后 ...

  7. python-jsonpath、findall返回值提取

    findall import re """ "d"表示取数字0-9, "D"表示不要数字, "w"在正则里面代 ...

  8. tomcat 启动一傘而过问题

    tomcat 启动一傘而过问题 D:\apache-tomcat-7.0.75\bin startup.bat打开记事本打开 第一行:设置启动环境变量JAVA_HOME,CATALINA_HOME S ...

  9. asp.net (web)选择文件夹 上传文件

    1 背景 用户本地有一份txt或者csv文件,无论是从业务数据库导出.还是其他途径获取,当需要使用蚂蚁的大数据分析工具进行数据加工.挖掘和共创应用的时候,首先要将本地文件上传至ODPS,普通的小文件通 ...

  10. POJ 2778 DNA Sequence ( AC自动机、Trie图、矩阵快速幂、DP )

    题意 : 给出一些病毒串,问你由ATGC构成的长度为 n 且不包含这些病毒串的个数有多少个 分析 : 这题搞了我真特么久啊,首先你需要知道的前置技能包括 AC自动机.构建Trie图.矩阵快速幂,其中矩 ...