题面

传送门

题目大意:

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. 基于VUE多人聊天项目

    项目背景 公司平台要做一个通讯系统,本来是java 来做的后面改前端+PHP来做,所以就用VUE来做这个了. github github地址 新人求star 技术栈 vue-axios vuex we ...

  2. ubuntu16.04 安装samba

    安装samba 1.更新当前软件 sudo apt-get upgrade sudo apt-get update sudo apt-get dist-upgrade 2.执行 sudo apt-ge ...

  3. 约数定理(two)

    筛约数个数和 理论基础: 1.对n质因数分解,n=p1^k1 * p2^k2 * p3^k3 …… 则n的约数个数为(k1+1)*(k2+1)*(k3+1)…… 2.线性筛素数时,用i和素数pj来筛掉 ...

  4. 手写Tomcat源码

    http://search.bilibili.com/all?keyword=%E6%89%8B%E5%86%99Tomcat%E6%BA%90%E7%A0%81 tomcat源码分析一:https: ...

  5. SQL 批量修改一个字段的值为另一个字段的值

    AND create_time BETWEEN '2016-07-25 14:30:00' AND '2016-07-25 15:20:28'; AND create_time BETWEEN '20 ...

  6. zepto-touch.js插件

    /* Zepto v1.2.0 - zepto event ajax form ie - zeptojs.com/license */(function(global, factory) { if ( ...

  7. db2数据库表操作错误SQL0668N Operation not allowed for reason code "1" on table "表". SQLSTATE=57016的解决方法

    错误sql Operation not allowed for reason code "1" on table "MARKET.PURE_USER".. SQ ...

  8. Linux内核调试方法总结之Kprobes

    Kprobes [用途][参考kernel/Documentation/kprobes.txt帮助文档] Kprobes是一个轻量级内核调试工具,同时又是其他一些更高级的内核调试工具(如perf和sy ...

  9. day03-Python基础

    1:函数 定义: 函数是指将一组语句的集合通过一个名字(函数名)封装起来,要想执行这个函数,只需调用其函数名即可 特性: 减少重复代码 使程序变的可扩展 使程序变得易维护 函数定义: def sayh ...

  10. fedora23禁用不需要的服务?--systemd服务单元?

    sign up: 签约; 登记, 注册. i'll sign up and go to the front and fight. he persuaded the company to sign up ...