https://www.lydsy.com/JudgeOnline/problem.php?id=4826

年少不知空间贵,相顾mle空流泪。

和上一道主席树求的东西差不多,求两种对

1. max(a[(i,j)])<min(a[i],a[j]),[i,j]这一对贡献p1.

2. max(a[(i,j)])在a[i],a[j]之间,[i,j]这一对贡献p2.

第一种和bzoj3956那道一样,但是因为是排列所以没必要去重了。

第二种同样是单调栈求lp,rp,每个位置的lp分别和[ i+1 , rp-1 ], rp分别和[ lp+1 , i-1 ], 构成了贡献p2的数对。

因此贡献为p2的数对要区间修改,因为方便(不用downdata)(其实是因为我抄的代码就是标记永久化)所以写了标记永久化,标记永久化真的挺好用的嘻嘻。

感觉写了这道题终于有点摸到主席树的门路了,其实就是找和维护两个区间进行限制的值(找和维护一个二维块),一层线段树一层前缀和(只是有传递性不一定是实际意义的和)。

 #include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
#define LL long long
const int maxn=;
LL n,m,p1,p2;
LL a[maxn]={},sta[maxn]={},tail=;
LL lp[maxn]={},rp[maxn]={},rt[maxn]={};
LL lc[maxn*]={},rc[maxn*]={},siz[maxn*]={},ad[maxn*]={},tot=;
struct nod{
LL x,l,r,v;
}e[maxn*];
LL cnt=;
bool mcmp(nod aa,nod bb){ return aa.x<bb.x; }
LL read(){
LL w=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){w=w*+ch-'';ch=getchar();}
return w*f;
}
void build(LL &x,LL y,LL l,LL r,LL z,LL zl,LL zr){
x=++tot;lc[x]=lc[y];rc[x]=rc[y];ad[x]=ad[y];siz[x]=siz[y]+z*(zr-zl+);
if(zl==l&&r==zr){ad[x]+=z;return;}
LL mid=(l+r)/;
if(zr<=mid)build(lc[x],lc[y],l,mid,z,zl,zr);
else if(zl>mid) build(rc[x],rc[y],mid+,r,z,zl,zr);
else {
build(lc[x],lc[y],l,mid,z,zl,mid);
build(rc[x],rc[y],mid+,r,z,mid+,zr);
}
}
LL getsum(LL x,LL y,LL l,LL r,LL zl,LL zr){
if(zl==l&&r==zr)return siz[y]-siz[x];
LL mid=(l+r)/,ans=(ad[y]-ad[x])*(zr-zl+);
if(zr<=mid)return ans+getsum(lc[x],lc[y],l,mid,zl,zr);
else if(zl>mid) return ans+getsum(rc[x],rc[y],mid+,r,zl,zr);
else {
return ans+getsum(lc[x],lc[y],l,mid,zl,mid)+getsum(rc[x],rc[y],mid+,r,mid+,zr);
}
}
//以上主席树
inline void fir(){
LL i;
a[]=a[n+]=(<<);
for(i=;i<=n;++i){
while(a[sta[tail]]<=a[i])tail--;
lp[i]=sta[tail];sta[++tail]=i;
}
sta[]=n+;tail=;
for(i=n;i>;--i){
while(a[sta[tail]]<=a[i])tail--;
rp[i]=sta[tail];sta[++tail]=i;
}
}
inline void init(LL x,LL l,LL r,LL v){
e[++cnt].x=x;e[cnt].l=l;e[cnt].r=r;e[cnt].v=v;
}
inline void fir2(){
LL i,j;
for(i=;i<=n;++i){
if(lp[i]!=&&rp[i]!=n+)init(lp[i],rp[i],rp[i],p1);
if(i<n)init(i,i+,i+,p1);
if(lp[i]!=&&rp[i]-i>)init(lp[i],i+,rp[i]-,p2);
if(rp[i]!=n+&&i-lp[i]>)init(rp[i],lp[i]+,i-,p2);
}
sort(e+,e+cnt+,mcmp);
for(i=j=;i<=n;++i){
rt[i]=rt[i-];
for(;e[j].x==i&&j<=cnt;++j)build(rt[i],rt[i],,n,e[j].v,e[j].l,e[j].r);
}
}
int main(){
LL i,x,y,ans=;
n=read();m=read();p1=read();p2=read();
for(i=;i<=n;++i)a[i]=read();
fir(); fir2();
for(i=;i<=m;++i){
x=read();y=read();
if(x>y)swap(x,y);
ans=getsum(rt[x-],rt[y],,n,x,y);
printf("%lld\n",ans);
}
return ;
}

BZOJ 4826: [Hnoi2017]影魔 单调栈 主席树的更多相关文章

  1. BZOJ 4826: [Hnoi2017]影魔 单调栈+可持久化线段树

    Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样 的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个 ...

  2. [BZOJ4826] [HNOI2017] 影魔 单调栈 主席树

    题面 因为是一个排列,所以不会有重复的.如果有重复就没法做了.一开始没有仔细看题目想了半天. 发现,如果是第一种情况,那么边界\(l\)和\(r\)就应该分别是整个区间的最大值和次大值. 然后,对于那 ...

  3. bzoj 4826: [Hnoi2017]影魔 [主席树 单调栈]

    4826: [Hnoi2017]影魔 题意:一个排列,点对\((i,j)\),\(p=max(i+1,j-1)\),若\(p<a_i,a_j\)贡献p1,若\(p\)在\(a_1,a_2\)之间 ...

  4. BZOJ 4826 [Hnoi2017]影魔 ——扫描线 单调栈

    首先用单调栈和扫描线处理出每一个数左面最近的比他大的数在$l[i]$,右面最近的比他大的数$r[i]$. 然后就可以考虑每种贡献是在什么时候产生的. 1.$(l[i],r[i])$产生$p1$的贡献 ...

  5. 【bzoj4826】[Hnoi2017]影魔 单调栈+可持久化线段树

    题目描述 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个灵魂,都有着自己 ...

  6. ●BZOJ 4826 [Hnoi2017]影魔

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=4826 题解: 主席树,单调栈 以前还没做过这种维护信息的题,感觉好奇妙. 每对相邻的两个数所 ...

  7. 【BZOJ4826】[Hnoi2017]影魔 单调栈+扫描线

    [BZOJ4826][Hnoi2017]影魔 Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝 ...

  8. bzoj 4826: [Hnoi2017]影魔【单调栈+树状数组+扫描线】

    参考:https://www.cnblogs.com/lcf-2000/p/6789680.html 这是一个相对码量少的做法,用到了区间修改区间查询的树状数组,详见:www.cnblogs.com/ ...

  9. BZOJ:4826: [Hnoi2017]影魔

    Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个灵 ...

随机推荐

  1. MVC Ajax Form & Ajax Valida(笔记)

    1.引入必要的文件 <script src=.min.js")" type="text/javascript"></script> &l ...

  2. docker 部署 portainer(http)

    =============================================== 2019/4/30_第6次修改                       ccb_warlock 更新 ...

  3. 24 The Go image package go图片包:图片包的基本原理

    The Go image package  go图片包:图片包的基本原理 21 September 2011 Introduction The image and image/color packag ...

  4. 移动端Touch事件

    案例1: <!doctype html> <html lang="en"> <head> <meta charset="UTF- ...

  5. Javascript之对象的创建

    面向对象语言有一个非常显著的标志,那就是它们都有类的概念,通过类之间的继承就可以达到任意创建具有相同属性方法的对象.而在ECMAScript中并没有类的概念,它把对象定义为:无序属性的集合,其属性包含 ...

  6. js中的call,apply,bind区别

    在JavaScript中,call.apply和bind是Function对象自带的三个方法,这三个方法的主要作用是改变函数中的this指向. call.apply.bind方法的共同点和区别:app ...

  7. (转载)solr实现满足指定距离范围条件的搜索

    配置schema.xml <?xml version="1.0" encoding="UTF-8" ?> <schema name=" ...

  8. Kubernetes 概述和搭建(多节点)

    一.Kubernetes整体概述和架构 Kubernetes是什么 Kubernetes是一个轻便的和可扩展的开源平台,用于管理容器化应用和服务.通过Kubernetes能够进行应用的自动化部署和扩缩 ...

  9. 定制Eclipse

    转载自http://chriszz.sinaapp.com 一般从Eclipse官网eclipse.org下载的,都是打包好的版本,比如标准版.jee版.java版.c++版.php版.测试版等.有时 ...

  10. C++中memcpy和memmove

    二者都是内存拷贝 memcpy内存拷贝,没有问题;memmove,内存移动?错,如果这样理解的话,那么这篇文章你就必须要好好看看了,memmove还是内存拷贝.那么既然memcpy和memmove二者 ...