【BZOJ4826】【HNOI2017】影魔
题意:
Description
Input
Output
共输出 m 行,每行一个答案,依次对应 m 个询问。
题解:
场上推出来了但是不会写……悲伤辣么大QAQ
题目中攻击力的计算方式如下:
$max\{k[s]|i<s<j\}\leq min\{k[i],k[j]\}$则会产生$p1$的贡献;
$min\{k[i],k[j]\}<max\{k[s]|i<s<j\}<max\{k[i],k[j]\}$则会产生$p2$的贡献;
即区间端点若是整个区间的最大值和次大值则会产生$p1$的贡献,若只有其中一个是最大值则会产生$p2$的贡献,特别地,区间$[i,i+1]$也会产生$p1$的贡献;
反过来考虑$i$对哪些点对有贡献,明显当且仅当$k[i]$是整个区间$[l,r]$的最大值时才会对点对$(l-1,r+1)$产生贡献。
题目中给出了关键条件是$k$数组是1~n的一个排列,所以不会有重复的数;
那么可以用单调栈求出$i$左边第一个比它大的位置$L[i]$和右边第一个位置$R[i]$,显然$k[i]$就是区间$[L[i]+1,R[i]-1]$中的最大值;
此时点对$(L[i],R[i])$会产生$p1$的贡献,点对$(L[i],x)(i<x<R[i])$和$(x,R[i])(L[i]<x<i)$会产生$p2$的贡献;
把点对看成二维平面上的点,这就变成了二维平面上某一个矩形内求和的问题,可以离线排序后用扫描线+树状数组解决。
但是考场上我不会写这个东西。。。下午去学了一发才会。。。
听说主席树也能做,但是我不会差分。。。
代码:
(不要在意里面的奇怪东西和卡榜用优读)
//大不列颠帝国尊贵的骑士王,圣剑Excalibur的持有者——阿尔托利亚·潘德拉贡 Arturia Pendragon
//永远是我的老婆,不接受反驳 #include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<queue>
#define inf 2147483647
#define eps 1e-9
#define lb(x) (x&-x)
using namespace std;
typedef long long ll;
struct task{
int id,z,l,r,v;
friend bool operator <(task a,task b){
return a.z<b.z;
}
}q1[],q2[];
int n,m,p1,p2,l,r,nw1=,nw2=,top=,tot1=,tot2=,num[],st[],L[],R[];
ll ans[],t1[],t2[];
char buffer[],*hd,*tl;
inline char Getchar(){
if(hd==tl){
int len=fread(buffer,,,stdin);
hd=buffer,tl=hd+len;
if(hd==tl)
return EOF;
}
return *hd++;
}
inline int rd(){
register int x=;
char c;
do c=Getchar();
while(!isdigit(c));
do{
x=(x<<)+(x<<)+(c^);
c=Getchar();
}while(isdigit(c));
return x;
}
void add(ll u,ll x){
ll uu=u;
for(;u<=n;u+=lb(u)){
t1[u]+=x;
t2[u]+=x*uu;
}
}
ll query(ll u){
ll ret=,uu=u;
for(;u;u-=lb(u)){
ret+=(uu+)*t1[u]-t2[u];
}
return ret;
}
int main(){
//scanf("%d%d%d%d",&n,&m,&p1,&p2);
n=rd(),m=rd(),p1=rd(),p2=rd();
num[]=num[n+]=inf;
st[top=]=;
for(int i=;i<=n;i++){
//scanf("%d",&num[i]);
num[i]=rd();
while(num[i]>num[st[top]])R[st[top--]]=i;
L[i]=st[top];
st[++top]=i;
}
while(top)R[st[top--]]=n+;
for(int i=;i<=m;i++){
//scanf("%d%d",&l,&r);
l=rd(),r=rd();
q1[++tot1]=(task){i,l-,l,r,-};
q1[++tot1]=(task){i,r,l,r,};
ans[i]=(ll)p1*(r-l);
}
for(int i=;i<=n;i++){
if(L[i]&&R[i]!=n+)q2[++tot2]=(task){,R[i],L[i],L[i],p1};
if(L[i]&&R[i]>i+)q2[++tot2]=(task){,L[i],i+,R[i]-,p2};
if(L[i]<i-&&R[i]!=n+)q2[++tot2]=(task){,R[i],L[i]+,i-,p2};
}
sort(q1+,q1+tot1+);
sort(q2+,q2+tot2+);
while(!q1[nw1].z)nw1++;
for(int i=;i<=n;i++){
for(;nw2<=tot2&&q2[nw2].z==i;nw2++){
add(q2[nw2].r+,-q2[nw2].v);
add(q2[nw2].l,q2[nw2].v);
}
for(;nw1<=tot1&&q1[nw1].z==i;nw1++){
ans[q1[nw1].id]+=(ll)q1[nw1].v*(query(q1[nw1].r)-query(q1[nw1].l-));
}
if(nw1>tot1)break;
}
for(int i=;i<=m;i++){
printf("%lld\n",ans[i]);
}
return ;
}
【BZOJ4826】【HNOI2017】影魔的更多相关文章
- [bzoj4826][Hnoi2017]影魔_单调栈_主席树
影魔 bzoj-4826 Hnoi-2017 题目大意:给定一个$n$个数的序列$a$,求满足一下情况的点对个数: 注释:$1\le n,m\le 2\cdot 10^5$,$1\le p1,p2\l ...
- [BZOJ4826][HNOI2017]影魔(主席树)
4826: [Hnoi2017]影魔 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 669 Solved: 384[Submit][Status][ ...
- bzoj4826 [Hnoi2017]影魔
Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝王.乞丐.奴隶.罪人,当然,还有英雄.每一个灵 ...
- bzoj千题计划196:bzoj4826: [Hnoi2017]影魔
http://www.lydsy.com/JudgeOnline/problem.php?id=4826 吐槽一下bzoj这道题的排版是真丑... 我还是粘洛谷的题面吧... 提供p1的攻击力:i,j ...
- BZOJ4826 [Hnoi2017]影魔 【线段树 + 单调栈】
题目链接 BZOJ4826 题解 蒟蒻智力水平捉急orz 我们会发现相邻的\(i\)和\(j\)贡献一定是\(p1\),可以很快算出来[然而我一开始忘了考虑调了半天] 我们现在只考虑不相邻的 我们只需 ...
- [BZOJ4826][HNOI2017]影魔 可持久化线段树
链接 题意:给你 \(1\) 到 \(n\) 的排列 \(k_1,k_2,\dots,k_n\) ,对 \(i,j (i<j)\)来说,若不存在 \(k_s (i<s<j)\) 大于 ...
- [BZOJ4826] [HNOI2017] 影魔 单调栈 主席树
题面 因为是一个排列,所以不会有重复的.如果有重复就没法做了.一开始没有仔细看题目想了半天. 发现,如果是第一种情况,那么边界\(l\)和\(r\)就应该分别是整个区间的最大值和次大值. 然后,对于那 ...
- 【BZOJ4826】[Hnoi2017]影魔 单调栈+扫描线
[BZOJ4826][Hnoi2017]影魔 Description 影魔,奈文摩尔,据说有着一个诗人的灵魂.事实上,他吞噬的诗人灵魂早已成千上万.千百年来,他收集了各式各样的灵魂,包括诗人.牧师.帝 ...
- bzoj 4826: [Hnoi2017]影魔 [主席树 单调栈]
4826: [Hnoi2017]影魔 题意:一个排列,点对\((i,j)\),\(p=max(i+1,j-1)\),若\(p<a_i,a_j\)贡献p1,若\(p\)在\(a_1,a_2\)之间 ...
- 4826: [Hnoi2017]影魔
4826: [Hnoi2017]影魔 https://lydsy.com/JudgeOnline/problem.php?id=4826 分析: 莫队+单调栈+st表. 考虑如何O(1)加入一个点,删 ...
随机推荐
- AVL树,红黑树,B树,B+树,Trie树都分别应用在哪些现实场景中?
AVL树: 最早的平衡二叉树之一.应用相对其他数据结构比较少.windows对进程地址空间的管理用到了AVL树. 红黑树: 平衡二叉树,广泛用在C++的STL中.如map和set都是用红黑树实现的. ...
- 11、E-commerce in Your Inbox:Product Recommendations at Scale-----产品推荐(prod2vec和user2vec)
一.摘要 本文提出一种方法,将神经语言模型应用在用户购买时间序列上,将产品嵌入到低维向量空间中.结果,具有相似上下文(即,其周围购买)的产品被映射到嵌入空间中附近的向量. 二.模型: 低维项目向量表示 ...
- Tensorflow学习笔记----基础(3)
目录: 一.TensorFlow的系统架构 二.TensorFlow的设计理念 三.TensorFlow的运行流程 四.TensorFlow的编程模型:边.节点.图.设备.变量.变量初始化.内核 五. ...
- 加减法计算器-java
由于经常进行较大数据的加减法计算,好多计算器都是转换成科学技术法的,所以自己用java写了一个 功能如下: 1,可以输入多个带千位分隔符的数字,进行加减法计算 2,结果展示带千位分隔符 3,结果展示不 ...
- CF1042F Leaf Sets (贪心+树上构造)
题目大意:给你一棵树,让你对叶节点分组,保证每组中,任意两个叶节点之间的距离不大于K,求最小的组数 手动yy的贪心竟然对的 对于每个节点,维护一个$ma[i]$,表示在$i$节点的子树内 未被分组的叶 ...
- vuex中mutations与actions的区别
要执行vuex中的函数,有两种方法: 1.commit,例如this.$store.commit("GETMODULESELECTLIST"); //mutations中的方法 2 ...
- mysql创建数据库并指定uft8编码
CREATE DATABASE IF NOT EXISTS ymk default character set utf8 COLLATE utf8_general_ci;
- vue解决跨域问题
vue解决跨域问题 vue跨域解决方法和小总结 vue项目中,前端与后台进行数据请求或者提交的时候,如果后台没有设置跨域,前端本地调试代码的时候就会报“No 'Access-Control-Allow ...
- CSLA框架的codesmith模板改造
一直有关注CSLA框架,最近闲来无事,折腾了下,在最新的r3054版本基础上修改了一些东西,以备自己用,有兴趣的园友可以下载共同研究 1.添加了默认的授权规则 如果是列表对象则生成列表权限,User的 ...
- 关于Segmentation fault错误
今天敲代码时候出现了Segmentation fault,在网上查了一些资料,基本上的原因是.非法的内存訪问. 比如数组的越界,在循环操作时循环变量的控制问题,也有字符串拷贝时长度溢出,指针指向了非法 ...