BZOJ 4826: [Hnoi2017]影魔 单调栈 主席树
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]影魔 单调栈 主席树的更多相关文章
- BZOJ 4826: [Hnoi2017]影魔 单调栈+可持久化线段树
Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样 的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个 ...
- [BZOJ4826] [HNOI2017] 影魔 单调栈 主席树
题面 因为是一个排列,所以不会有重复的.如果有重复就没法做了.一开始没有仔细看题目想了半天. 发现,如果是第一种情况,那么边界\(l\)和\(r\)就应该分别是整个区间的最大值和次大值. 然后,对于那 ...
- bzoj 4826: [Hnoi2017]影魔 [主席树 单调栈]
4826: [Hnoi2017]影魔 题意:一个排列,点对\((i,j)\),\(p=max(i+1,j-1)\),若\(p<a_i,a_j\)贡献p1,若\(p\)在\(a_1,a_2\)之间 ...
- BZOJ 4826 [Hnoi2017]影魔 ——扫描线 单调栈
首先用单调栈和扫描线处理出每一个数左面最近的比他大的数在$l[i]$,右面最近的比他大的数$r[i]$. 然后就可以考虑每种贡献是在什么时候产生的. 1.$(l[i],r[i])$产生$p1$的贡献 ...
- 【bzoj4826】[Hnoi2017]影魔 单调栈+可持久化线段树
题目描述 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个灵魂,都有着自己 ...
- ●BZOJ 4826 [Hnoi2017]影魔
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=4826 题解: 主席树,单调栈 以前还没做过这种维护信息的题,感觉好奇妙. 每对相邻的两个数所 ...
- 【BZOJ4826】[Hnoi2017]影魔 单调栈+扫描线
[BZOJ4826][Hnoi2017]影魔 Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝 ...
- bzoj 4826: [Hnoi2017]影魔【单调栈+树状数组+扫描线】
参考:https://www.cnblogs.com/lcf-2000/p/6789680.html 这是一个相对码量少的做法,用到了区间修改区间查询的树状数组,详见:www.cnblogs.com/ ...
- BZOJ:4826: [Hnoi2017]影魔
Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个灵 ...
随机推荐
- post请求远程url 报错“基础连接已经关闭...Authentication.AuthenticationException...远程证书无效”解决方案
当我们有时用代码编写post请求url远程地址会报“基础连接已经关闭: 未能为 SSL/TLS 安全通道建立信任关系. ---> System.Security.Authentication.A ...
- mysqli链接数据库示例代码
$mysqli = new mysqli("localhost", "数据库用户名", "数据库密码", "数据库名称" ...
- 四、Springboot Debug调试
描述: 在使用maven插件执行spring-boot:run进行启动的时候,如果设置的断点进不去,要进行以下的设置. 1.添加jvm参数配置 在spring-boot的maven插件加上jvmArg ...
- Python开发环境(2):启动Eclipse时检测到PYTHONPATH发生改变
OS:Windows 10家庭中文版,Eclipse:Oxygen.1a Release (4.7.1a),PyDev:6.3.2 4月25日,在Eclipse上安装了PyDev(前面博文有记录),并 ...
- linux(vi)常用命令
常用操作 系统命令 查看主机名 hostname 修改主机名(重启后无效) hostname yang 修改主机名(重启后永久生效) vi /ect/sysconfig/network 修改IP(重启 ...
- 如何在线预览github上的html页面?
可以通过http://htmlpreview.github.io/这个网站,直接在线预览html页面. ↓ ↓ 可以发现:这个网站直接将github上的页面地址当做参数来传递.
- 【笔记】jQuery插件开发指南
原文链接:http://www.cnblogs.com/Wayou/p/jquery_plugin_tutorial.html (有部分增删和修改) jQuery插件开发模式 软件开发过程中是需要一定 ...
- 在VirtualBox虚拟机中安装Centos操作系统怎么与本地XShell远程连接
问题: 在VirtualBox安装好了CentOS操作系统后,我们怎么才可以用XSell连接虚拟机中的CentOS呢? 答案: (1)在windows下用cmd--ipconfig查看VirtualB ...
- (三)HtmlUnit 实践
第一节: htmlunit 爬取百度云资源
- MySql数据库 主从复制/共享 报错
从 获取不到 共享主的数据, 错误信息: Waiting for master to send event 解决方案: // 1. 从V表获取PrNo的数据 select * from Vendor_ ...