【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)\) 大于 ...
随机推荐
- AC日记——[NOI2006]最大获利 bzoj 1497
1497 思路: 最小割: 来,上代码: #include <cstdio> #include <cstring> #include <iostream> #inc ...
- (6)python tkinter-容器、子窗体
Frame f = tkinter.Frame(width=380, height=270, bg='white').pack() LabelFrame f = tkinter.LabelFrame( ...
- Android 日期对话框DatePickerDialog
<?xml version="1.0" encoding="utf-8"?> <LinearLayout android:id="@ ...
- Scut游戏服务器引擎之Unity3d接入
Scut提供Unity3d Sdk包,方便开发人员快速与Scut游戏服务器对接: 先看Unity3d示例如下: 启动Unity3d项目 打开Scutc.svn\SDK\Unity3d\Assets目录 ...
- EasyMvc入门教程-基本控件说明(8)提醒导航
提醒导航顾名思义就是提醒大家注意某些文字了..请看下面的例子: 实现代码如下: @Html.Q().BlockRemind().Text("我可以作为提醒使用") 有的同学会说:这 ...
- H5 性能调优 工具
1.阿里测:http://www.alibench.com 2.奇云测:http://ce.cloud.360.cn 3.百度应用性能检测中心:http://apm.baidu.com 推荐理由:这3 ...
- 解释一下Windows dos中的符号
容许我放一段Windows的批处理: sc <server> [command] [service name] <option1> <option2>... < ...
- GUID概念
GUID概念 GUID: 即Globally Unique Identifier(全球唯一标识符) 也称作 UUID(Universally Unique IDentifier) . GUID是 ...
- MYSQL 的optimize怎么用
当对表有大量的增删改操作时,需要用optimize对表进行优化.可以减少空间与提高I/O性能,命令optimize table tablename;假如有foo表且存储引擎为MyISAM. mysql ...
- 翻翻git之---一个丰富的通知工具类 NotifyUtil
转载请注明出处王亟亟的大牛之路 P1(废话板块.今天还加了个小广告) 昨天出去浪,到家把麦麦当当放出来玩一会就整到了12点多..早上睡过头了. .简直心酸. ... 近期手头上有一些职位能够操作,然后 ...