CF997E Good Subsegments

给你一个长度为\(n\)的排列 \(P\),定义一段子区间是好的,当且仅当这个子区间内的值构成了连续的一段。例如对于排列\(\{1,3,2\}\),\([1,1],[2,2],[3,3],[2,3],[1,3]\)是好的区间。

共\(q\)次询问,每次询问\(L,R\), 求有多少\(L≤l≤r≤R\),满足\([l,r]\)是好的区间。\(1≤n,q≤1.2×10^5\).


积累一个套路:对于询问区间子序列的信息,可以离线移动右指针,类似扫描线一样计算贡献。

注意到若区间\([l,r]\)满足\(max-min=r-l\),则这个区间是好的区间。

于是在当前指针\(r\)下维护一个\((max-min)-(r-l)\)的最小值以及最小值个数。

然后维护一个答案\(sum\),代表某个区间最小值的贡献(注意这时候不需要维护是不是\(0\),我们只需要在最后累计答案的时候在树根判断就可以了,这个题根一定为\(0\),所以没有判)

但是发现这样不行,我们只是维护了一个右端点为\(r\)的区间的贡献,于是维护一个\(dev\)贡献数组,维护历史出现的最小值出现的贡献。

具体点说,\(dev\)也相当于一个\(tag\),表示这个区间当前的答案需要被贡献到\(sum\),不管它之后怎么样。

然后区间\(max,min\)的相关东西用单调栈额外维护,复杂度是均摊的\(\log n\)

一次操作的大体流程:

  1. 对整个区间打一个右移标记,即区间最小值减\(1\)
  2. 维护单调栈并更新一下\(max,min\)
  3. 打上历史更新\(tag\)
  4. 进行询问

一些细节:初始时每个区间的\(mi\)是它自己的左端点,因为每次是对整颗树的区间打的标记,会减去很多次\(1\),下放标记的时候更新答案看是不是跟自己一样,如果子区间最小值跟自己一样才进行更新。


Code:

#include <cstdio>
#include <algorithm>
#define ll long long
const int N=120010;
using std::min;
int mi[N<<2],ct[N<<2],mitg[N<<2],dev[N<<2];
//最小值,最小值次数,最小值tag,历史贡献tag
ll ans[N],sum[N<<2];
int sta1[N],tot1,sta2[N],tot2;
int n,m,p[N];
struct node
{
int id,l,r;
bool friend operator <(node n1,node n2){return n1.r<n2.r;}
}q[N];
#define ls id<<1
#define rs id<<1|1
void build(int id,int l,int r)
{
mi[id]=l,ct[id]=1;int mid=l+r>>1;
if(l^r) build(ls,l,mid),build(rs,mid+1,r);
}
void addmi(int id,int d){mi[id]+=d,mitg[id]+=d;}
void addde(int id,int d){sum[id]+=1ll*ct[id]*d,dev[id]+=d;}
void updata(int id)
{
mi[id]=min(mi[ls],mi[rs]),ct[id]=0;
ct[id]=mi[id]==mi[ls]?ct[id]+ct[ls]:ct[id];
ct[id]=mi[id]==mi[rs]?ct[id]+ct[rs]:ct[id];
sum[id]=sum[ls]+sum[rs];
}
void pushdown(int id)
{
if(mitg[id]) addmi(ls,mitg[id]),addmi(rs,mitg[id]),mitg[id]=0;
if(dev[id])
{
if(mi[id]==mi[ls]) addde(ls,dev[id]);
if(mi[id]==mi[rs]) addde(rs,dev[id]);
dev[id]=0;
}
}
void change(int id,int L,int R,int l,int r,int d)
{
if(l==L&&r==R){addmi(id,d);return;}
int Mid=L+R>>1;pushdown(id);
if(r<=Mid) change(ls,L,Mid,l,r,d);
else if(l>Mid) change(rs,Mid+1,R,l,r,d);
else change(ls,L,Mid,l,Mid,d),change(rs,Mid+1,R,Mid+1,r,d);
updata(id);
}
ll query(int id,int L,int R,int l,int r)
{
if(l==L&&r==R)return sum[id];
int Mid=L+R>>1;pushdown(id);
if(r<=Mid) return query(ls,L,Mid,l,r);
else if(l>Mid) return query(rs,Mid+1,R,l,r);
else return query(ls,L,Mid,l,Mid)+query(rs,Mid+1,R,Mid+1,r);
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++) scanf("%d",p+i);
scanf("%d",&m);
for(int i=1;i<=m;i++) scanf("%d%d",&q[i].l,&q[i].r),q[i].id=i;
std::sort(q+1,q+1+m);
build(1,1,n);
for(int j=1,i=1;i<=n;i++)
{
addmi(1,-1);
while(tot1&&p[sta1[tot1]]<=p[i])
{
change(1,1,n,sta1[tot1-1]+1,sta1[tot1],p[i]-p[sta1[tot1]]);
--tot1;
}
sta1[++tot1]=i;
while(tot2&&p[sta2[tot2]]>=p[i])
{
change(1,1,n,sta2[tot2-1]+1,sta2[tot2],p[sta2[tot2]]-p[i]);
--tot2;
}
sta2[++tot2]=i;
addde(1,1);
while(q[j].r==i) ans[q[j].id]=query(1,1,n,q[j].l,q[j].r),++j;
}
for(int i=1;i<=m;i++) printf("%lld\n",ans[i]);
return 0;
}

2018.12.25

CF 997E 解题报告的更多相关文章

  1. 12.27 cf div3 解题报告

    12.27 cf div3 解题报告 wxy.wxy,带上分拉,全场做了个无脑小白 比赛场地 A: T1,跟着模拟就好了 B: sort一遍之后 去除的数一定是a[1]或者a[n] 比较去除谁小就输出 ...

  2. CF 1087解题报告

    cf解题报告 记录一下吧 做出:T1 rating :-97 想起几个月前做不出T1还是有点小搞笑呀2333 T1 双指针+特判 T2 发现k特别小,枚举剩余系 还要判断是否是能被n整除 移项发现可以 ...

  3. CF 1215解题报告

    T1 偶数输出n/2 奇数输出(n-1)/2即可 T2 判断是不是回文 不是直接输出子串 是回文继续判断 如果他前(len+1)/2内没有相同 输出-1 其他的 交换不同字符,输出子串 T3 贪心+二 ...

  4. 冲刺Noip2017模拟赛3 解题报告——五十岚芒果酱

    题1  素数 [问题描述] 给定一个正整数N,询问1到N中有多少个素数. [输入格式]primenum.in 一个正整数N. [输出格式]primenum.out 一个数Ans,表示1到N中有多少个素 ...

  5. CF Round #600 (Div 2) 解题报告(A~E)

    CF Round #600 (Div 2) 解题报告(A~E) A:Single Push 采用差分的思想,让\(b-a=c\),然后观察\(c\)序列是不是一个满足要求的序列 #include< ...

  6. Codeforces Round #382 (Div. 2) 解题报告

    CF一如既往在深夜举行,我也一如既往在周三上午的C++课上进行了virtual participation.这次div2的题目除了E题都水的一塌糊涂,参赛时的E题最后也没有几个参赛者AC,排名又成为了 ...

  7. Mutual Training for Wannafly Union #1解题报告

    ---恢复内容开始--- q神等人组织的vjudge上的多校训练,题目基本上都来自于CF,#1是上周进行的,参加后感觉收获很多,因为上周准备期中比较忙,解题报告现在补上. 比赛地址(兼题目地址) A题 ...

  8. Codeforces Round #378 (Div. 2) D题(data structure)解题报告

    题目地址 先简单的总结一下这次CF,前两道题非常的水,可是第一题又是因为自己想的不够周到而被Hack了一次(或许也应该感谢这个hack我的人,使我没有最后在赛后测试中WA).做到C题时看到题目情况非常 ...

  9. Codeforces Round #277.5 解题报告

    又熬夜刷了cf,今天比正常多一题.比赛还没完但我知道F过不了了,一个半小时贡献给F还是没过--应该也没人Hack.写写解题报告吧= =. 解题报告例如以下: A题:选择排序直接搞,由于不要求最优交换次 ...

随机推荐

  1. python包管理工具pip

    你可以使用一个名为 pip 的程序来安装.升级和移除软件包.默认情况下 pip 将从 Python Package Index <https://pypi.org> 安装软件包.你可以在浏 ...

  2. [codeForce-1006C]-Three Parts of the Array (简单题)

    You are given an array d1,d2,…,dnd1,d2,…,dn consisting of nn integer numbers. Your task is to split ...

  3. CHAPTER 25 The Greatest Show on Earth 第25章 地球上最壮观的演出

    CHAPTER 25 The Greatest Show on Earth 第25章 地球上最壮观的演出 Go for a walk in the countryside and you will f ...

  4. linux go环境安装

    方法一 这次将源码包安装的目录是是/root下. 1.官网下载源码包. 官网链接:https://golang.org/dl/   wget https://storage.googleapis.co ...

  5. Python基础系列讲解——继承派生和组合的概念剖析

    Python作为一门面向对象的语言,它的面向对象体系中主要存在这么两种关系,一个是“类”和“实例”的关系,另一个是“父类”和“子类”的关系. 所谓“类”是从一堆对象中以抽象的方式把相同的特征归类得到的 ...

  6. centos 切换用户显示bash-4.2$,不显示用户名路径的问题

    原文链接: http://blog.csdn.net/testcs_dn/article/details/70482468

  7. 2017-2018-2 1723 『Java程序设计』课程 结对编程练习-四则运算-最后阶段

    2017-2018-2 1723 『Java程序设计』课程 结对编程练习-四则运算-最后阶段 最后的一周,时间越来越紧张,因为之前的拖沓和一些事情的耽误,导致了如今的紧张,这一周应该是我们小组效率最高 ...

  8. bata1

    目录 组员情况 组员1(组长):胡绪佩 组员3:庄卉 组员4:家灿 组员5:凯琳 组员6:翟丹丹 组员7:何家伟 组员8:政演 组员9:黄鸿杰 组员10:刘一好 组员11:何宇恒 展示组内最新成果 团 ...

  9. Leetcode题库——8.字符串转为整数【##】

    @author: ZZQ @software: PyCharm @file: myAtoi.py @time: 2018/9/20 20:54 要求:实现 atoi,将字符串转为整数. 1)根据需要丢 ...

  10. DFS--障碍在指定时间会消失

    哈利被困在了一个魔法花园里.魔法花园是一个 N*M 的矩形,在其中有着许多植物, 这些植物会在时刻 K 的倍数消失. 哈利每单位时间都会选择上.下.左.右四 个方向的其中一个进行移动. #includ ...