【BZOJ4826】[Hnoi2017]影魔 单调栈+扫描线
【BZOJ4826】[Hnoi2017]影魔
Description
Input
Output
Sample Input
7 9 5 1 3 10 6 8 2 4
1 7
1 9
1 3
5 9
1 5
Sample Output
39
4
13
16
题解:用单调栈求出每个数左边和右边第一个比它大的数的位置,设其为l[i],r[i]。那么这样的点对:
(l[i],r[i]),(i,i+1)提供p1攻击力
(l[i]...i-1,r[i]),(l[i],i+1...r[i])提供p2攻击力
那么这就变成了在二维平面中,一些线段和点有权值,求矩形内的权值和。可以用扫描线解决(+区间修改树状数组)。
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int maxn=200010;
typedef long long ll;
int n,m,n1,n2,m1,top;
int v[maxn],lm[maxn],rm[maxn],st[maxn];
ll p1,p2;
ll ans[maxn];
struct sag
{
int sx,sl,sr,sv;
}s1[maxn<<1],s2[maxn<<1];
struct QUERY
{
int qx,ql,qr,org,qv;
}q[maxn<<1];
struct BIT
{
ll s[maxn];
void init(){memset(s,0,sizeof(s));}
void updata(int x,ll val)
{
for(int i=x;i<=n;i+=i&-i) s[i]+=val;
}
ll query(int x)
{
ll ret=0;
for(int i=x;i;i-=i&-i) ret+=s[i];
return ret;
}
}sa,sb;
void modify(int x,ll val)
{
if(x<=0) return ;
sa.updata(x,val),sb.updata(x,val*x);
}
ll ask(int x)
{
return (sa.query(n)-sa.query(x))*x+sb.query(x);
}
int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<'0'||gc>'9') {if(gc=='-')f=-f; gc=getchar();}
while(gc>='0'&&gc<='9') ret=ret*10+gc-'0',gc=getchar();
return ret*f;
}
bool cmps(sag a,sag b)
{
return a.sx<b.sx;
}
bool cmpq(QUERY a,QUERY b)
{
return a.qx<b.qx;
}
int main()
{
n=rd(),m=rd(),p1=rd(),p2=rd();
int i,j;
for(i=1;i<=n;i++) v[i]=rd();
for(top=0,i=1;i<=n;i++)
{
while(top&&v[st[top]]<v[i]) rm[st[top--]]=i;
lm[i]=st[top],st[++top]=i;
}
for(i=1;i<=n;i++)
{
if(lm[i]&&rm[i]) s1[++n1].sx=lm[i],s1[n1].sl=s1[n1].sr=rm[i],s1[n1].sv=p1;
if(i<n) s2[++n2].sx=i+1,s2[n2].sl=s2[n2].sr=i,s2[n2].sv=p1;
if(!rm[i]) rm[i]=n+1;
if(lm[i]&&rm[i]>i+1) s1[++n1].sx=lm[i],s1[n1].sl=i+1,s1[n1].sr=rm[i]-1,s1[n1].sv=p2;
if(rm[i]<=n&&lm[i]<i-1) s2[++n2].sx=rm[i],s2[n2].sl=lm[i]+1,s2[n2].sr=i-1,s2[n2].sv=p2;
}
sort(s1+1,s1+n1+1,cmps),sort(s2+1,s2+n2+1,cmps);
for(i=1;i<=m;i++)
{
q[i].qx=q[i].ql=q[i+m].ql=rd(),q[i+m].qx=q[i].qr=q[i+m].qr=rd();
q[i].qx--,q[i].org=i=q[i+m].org=i,q[i].qv=-1,q[i+m].qv=1;
}
sort(q+1,q+2*m+1,cmpq);
for(i=j=1;i<=2*m;i++)
{
if(!q[i].qx) continue;
for(;j<=n1&&s1[j].sx<=q[i].qx;j++) modify(s1[j].sl-1,-s1[j].sv),modify(s1[j].sr,s1[j].sv);
ans[q[i].org]+=q[i].qv*(ask(q[i].qr)-ask(q[i].ql-1));
}
sa.init(),sb.init();
for(i=j=1;i<=2*m;i++)
{
if(!q[i].qx) continue;
for(;j<=n2&&s2[j].sx<=q[i].qx;j++) modify(s2[j].sl-1,-s2[j].sv),modify(s2[j].sr,s2[j].sv);
ans[q[i].org]+=q[i].qv*(ask(q[i].qr)-ask(q[i].ql-1));
}
for(i=1;i<=m;i++) printf("%lld\n",ans[i]);
return 0;
}
【BZOJ4826】[Hnoi2017]影魔 单调栈+扫描线的更多相关文章
- [BZOJ4826] [HNOI2017] 影魔 单调栈 主席树
题面 因为是一个排列,所以不会有重复的.如果有重复就没法做了.一开始没有仔细看题目想了半天. 发现,如果是第一种情况,那么边界\(l\)和\(r\)就应该分别是整个区间的最大值和次大值. 然后,对于那 ...
- 【bzoj4826】[Hnoi2017]影魔 单调栈+可持久化线段树
题目描述 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个灵魂,都有着自己 ...
- BZOJ 4826: [Hnoi2017]影魔 单调栈+可持久化线段树
Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样 的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个 ...
- BZOJ 4826: [Hnoi2017]影魔 单调栈 主席树
https://www.lydsy.com/JudgeOnline/problem.php?id=4826 年少不知空间贵,相顾mle空流泪. 和上一道主席树求的东西差不多,求两种对 1. max(a ...
- [bzoj4826][Hnoi2017]影魔_单调栈_主席树
影魔 bzoj-4826 Hnoi-2017 题目大意:给定一个$n$个数的序列$a$,求满足一下情况的点对个数: 注释:$1\le n,m\le 2\cdot 10^5$,$1\le p1,p2\l ...
- P3722 [AH2017/HNOI2017]影魔(单调栈+扫描线+线段树)
题面传送门 首先我们把这两个贡献翻译成人话: 区间 \([l,r]\) 产生 \(p_1\) 的贡献当且仅当 \(a_l,a_r\) 分别为区间 \([l,r]\) 的最大值和次大值. 区间 \([l ...
- bzoj4826 [Hnoi2017]影魔
Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个灵 ...
- 【CF815D】Karen and Cards 单调栈+扫描线
[CF815D]Karen and Cards 题意:一张卡片有三个属性a,b,c,其上限分别为A,B,C,现在有n张卡片,定义一张卡片能打败另一张卡片当且仅当它的至少两项属性要严格大于另一张的对应属 ...
- [BZOJ4826][HNOI2017]影魔 可持久化线段树
链接 题意:给你 \(1\) 到 \(n\) 的排列 \(k_1,k_2,\dots,k_n\) ,对 \(i,j (i<j)\)来说,若不存在 \(k_s (i<s<j)\) 大于 ...
随机推荐
- RadioGroup动态添加RadioButton,并且获得事件
由于有许多的RadioButton是动态的,不是固定的一些,所以需要在代码中,动态的添加到RadioGroup中,下面是我的实现方法. 1.添加RadioButton到RadioGroup中 Radi ...
- AC日记——[中山市选2009]谁能赢呢? bzoj 2463
2463 思路: 博弈: 把先手和后手的走的两个格子看做一个1*2的方格: 如果n为偶数,那么棋盘一定可以被1*2的方格覆盖: 前端为先手,后端为后手: 那么,当还剩下一个1*2的方格时,先手一定可以 ...
- Codeforces 898 C.Phone Numbers-STL(map+set+vector)
C. Phone Numbers time limit per test 2 seconds memory limit per test 256 megabytes input standard ...
- HDU5877 Weak Pair
题目链接 Weak Pair 题意十分明确, 就是求出符合题意的有序点对个数. 首先对ai离散,离散之后的结果用rk[i]表示,然后进行二分预处理得到f[i],其中f[i]的意义为:其他的点和i这个节 ...
- 身份识别协议枚举工具ident-user-enum
身份识别协议枚举工具ident-user-enum 身份识别协议(Ident protocol,IDENT)是一种Internet协议,用于识别使用特定TCP端口的用户身份.服务器开启该服务后,会 ...
- iOS for循环创建button,button的宽度依据上面的文字来自适应.
近期须要使用一个标签页,寻思自己写一个demo. 标签的大小依据上面的文字来自适应大小,须要依据后台返回的数据自己主动换行.没有加入 NSArray *arr = @[@"无知", ...
- cocos2d-x 3.0游戏实例学习笔记 《跑酷》 第三步---主角开跑&同一时候带着刚体
说明:这里是借鉴:晓风残月前辈的博客.他是将泰然网的跑酷教程.用cocos2d-x 2.X 版本号重写的,眼下我正在学习cocos2d-X3.0 于是就用cocos2d-X 3.0重写,并做相关笔记 ...
- DbVisualizer9.0.6破解版下载、破解方法以及补丁
DbVisualizer破解方法如下:第一步:把下载的dbvis.puk文件,替换掉安装目录“D:\Program Files\DbVisualizer-6.0.12\lib”下dbvis.jar里面 ...
- 2016.7.14 generator基于注解和基于xml自动生成代码的区别
1.generator配置文件generatorConfig.xml的区别 2.生成代码的区别 注:二者的实体类都一样. (1)基于XML 生成的文件有: 后面省略. 也就是说,基于xml的方式,是要 ...
- 从头认识java-14.2 进一步了解数组
这一章节我们来全面了解一下数组. 1.数组在初始化之前.我们不能使用他的引用来做不论什么事情. package com.ray.ch14; public class Test { public sta ...