http://codeforces.com/contest/193/problem/D

题意:

给一个1~n的排列,在这个排列中选出两段区间,求使选出的元素排序后构成公差为1的等差数列的方案数。

换个角度思考问题,题意转化为存在多少对[L,R] ,(R>L),满足将值为[L,R]的区间染色后,所得区间数<=2

假设现在已知[L,R]的染色情况,看将值为L-1的位置染色后,区间数量的变化

若L-1左右两边都没有染色,区间数量+1

若L-1左右两边有一边染了色,区间数量不变

若L-1左右两边都染色了,区间数量-1

这样就有了枚举L R 的 n^2 做法

令f[i]表示当左端点为L,有端点为i时区间的数量

从大到小枚举L

考虑由[L,m]   m∈[L+1,n]  到 [L-1,m]  m∈[L,n] 时

f[i] i∈[L-1,n]的变化

设L-1 左右两边的数分别为x和y,且x<y

A、L-1左右两边都没有染色,即x<y<L-1,

染上L-1后会使区间数+1,即f[i]加1 ,i∈[L-1,n]

B、L-1左右两边有一边染色,即x<L-1<y,(y的那一边染色)

若染色的区间原本不包含y,染上L-1后会使区间数+1,即f[i]加1,i∈[L-1,y-1]

若染色的区间原本包含y,L-1与y相连,染上L-1后区间数不变

C、L-1左右两边都染色了,即L-1<x<y

若染色的区间原本不包含x,也不包含y,染上L-1后会使区间数+1,即f[i]加1,i∈[L-1,x-1]

若染色的区间原本只包含其中一个(只包含y),染山L-1后区间数不变

若染色的区间原本包含x和y,染上L-1后,两边的区间相连,区间数-1,即f[i]减1,i∈[y,n]

用线段树维护f[i]

这就变成了线段树的区间+1,区间-1,查询区间内f[i]<=2的数的个数

维护区间最小值mi[i],等于最小值的个数tot[i],等于最小值+1的个数tot1[i]

答案由两部分组成:

1、mi[i]<=2,ans+=tot[i]

2、mi[i]==1,ans+=tot1[i]

#include<cstdio>
#include<iostream>
#include<algorithm> using namespace std; #define N 300001 int a[N+],b[N+]; int tag[N<<];
int tot[N<<],tot1[N<<];
int mi[N<<]; long long ans=; void read(int &x)
{
x=; char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) { x=x*+c-''; c=getchar(); }
} void build(int k,int l,int r)
{
tot[k]=r-l+;
if(l==r) return;
int mid=l+r>>;
build(k<<,l,mid);
build(k<<|,mid+,r);
} void down(int k)
{
tag[k<<]+=tag[k];
tag[k<<|]+=tag[k];
mi[k<<]+=tag[k];
mi[k<<|]+=tag[k];
tag[k]=;
} void update(int k)
{
mi[k]=min(mi[k<<],mi[k<<|]);
tot[k]=tot[k<<]*(mi[k<<]==mi[k])+tot[k<<|]*(mi[k<<|]==mi[k]);
tot1[k]=tot1[k<<]*(mi[k<<]==mi[k])+tot1[k<<|]*(mi[k<<|]==mi[k]);
tot1[k]+=tot[k<<]*(mi[k<<]==mi[k]+)+tot[k<<|]*(mi[k<<|]==mi[k]+);
} void change(int k,int l,int r,int opl,int opr,int w)
{
if(l>=opl && r<=opr)
{
mi[k]+=w;
tag[k]+=w;
return;
}
if(tag[k]) down(k);
int mid=l+r>>;
if(opl<=mid) change(k<<,l,mid,opl,opr,w);
if(opr>mid) change(k<<|,mid+,r,opl,opr,w);
update(k);
} void query(int k,int l,int r,int opl,int opr)
{
if(l>=opl && r<=opr)
{
ans+=tot[k]*(mi[k]<=)+tot1[k]*(mi[k]==);
return;
}
if(tag[k]) down(k);
int mid=l+r>>;
if(opl<=mid) query(k<<,l,mid,opl,opr);
if(opr>mid) query(k<<|,mid+,r,opl,opr);
} int main()
{
int n,m;
read(n);
int x,y;
for(int i=;i<=n;++i) read(x),a[x]=i;
build(,,n);
for(int i=n;i;--i)
{
b[a[i]]=i;
x=b[a[i]-];
y=b[a[i]+];
if(x>y) swap(x,y);
if(x)
{
change(,,n,y,n,-);
change(,,n,i,x-,);
}
else if(y) change(,,n,i,y-,);
else change(,,n,i,n,);
//long long last=ans;
query(,,n,i,n);
//cout<<i<<' '<<ans-last<<'\n';
}
cout<<ans-n;
}

Codeforces 193 D. Two Segments的更多相关文章

  1. Codeforces 1108E2 Array and Segments (Hard version) 差分, 暴力

    Codeforces 1108E2 E2. Array and Segments (Hard version) Description: The only difference between eas ...

  2. Codeforces 1108E2 Array and Segments (Hard version)(差分+思维)

    题目链接:Array and Segments (Hard version) 题意:给定一个长度为n的序列,m个区间,从m个区间内选择一些区间内的数都减一,使得整个序列的最大值减最小值最大. 题解:利 ...

  3. Codeforces 895.B XK Segments

    B. XK Segments time limit per test 1 second memory limit per test 256 megabytes input standard input ...

  4. 【CodeForces】899 E. Segments Removal

    [题目]E. Segments Removal [题意]给定n个数字,每次操作删除最长的连续相同数字(等长删最左),求全部删完的最少次数.n<=2*10^6,1<=ai<=10^9. ...

  5. 『ACM C++』 Codeforces | 1066A - Points in Segments

    大一生活真 特么 ”丰富多彩“ ,多彩到我要忙到哭泣,身为班长,很多班级的事情需要管理,也是,什么东西都得体验学一学,从学生会主席.团委团总支.社团社长都体验过一番了,现在差个班长也没试过,就来体验了 ...

  6. codeforces 652D D. Nested Segments(离散化+sort+树状数组)

    题目链接: D. Nested Segments time limit per test 2 seconds memory limit per test 256 megabytes input sta ...

  7. Codeforces 620F Xors on Segments(暴力+DP)

    题目链接 Xors on Segments 预处理出$x[i]$ $=$ $1$ $xor$ $2$ $xor$ $3$ $xor$ $……$ $xor$ $i$ 话说这题$O(n^{2})$居然能过 ...

  8. codeforces#1108E2. Array and Segments (线段树+扫描线)

    题目链接: http://codeforces.com/contest/1108/problem/E2 题意: 给出$n$个数和$m$个操作 每个操作是下标为$l$到$r$的数减一 选出某些操作,使$ ...

  9. Codeforces 429E - Points and Segments(欧拉回路)

    Codeforces 题面传送门 & 洛谷题面传送门 果然我不具备融会贯通的能力/ll 看到这样的设问我们可以很自然地联想到这道题,具体来说我们可以通过某种方式建出一张图,然后根据" ...

随机推荐

  1. Gearman研习笔记(1) ------ 官网介绍要点摘录

    之前的项目里使用过消息中间件(公司提供的MQ服务)来做分发,因为MQ是基于消息的,并不是专业的任务分发器,在一些复杂场景上使用起来并不恰当. 后来听组长说了下Gearman(听名字还以为是Ironma ...

  2. 【BZOJ2599】Race(点分治)

    [BZOJ2599]Race(点分治) 题面 BZOJ权限题,洛谷 题解 好久没写过点分治了... 在ppl的帮助下终于想起来了 orz ppl 首先回忆一下怎么求有没有正好是\(K\)的路径 维护一 ...

  3. 【BZOJ4736】温暖会指引我们前行(Link-Cut Tree)

    [BZOJ4736]温暖会指引我们前行(Link-Cut Tree) ##题面 神TM题面是UOJ的 题解 LCT傻逼维护最大生成树 不会的可以去做一做魔法森林 #include<iostrea ...

  4. Scala编程快速入门系列(一)

    目    录 一.Scala概述 二.Scala数据类型 三.Scala函数 四.Scala集合 五.Scala伴生对象 六.Scala trait 七.Actor 八.隐式转换与隐式参数 九.Sca ...

  5. node.js-v6新版安装过程

    1.Node.js简介 简单的说 Node.js 就是运行在服务端的 JavaScript.Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境.Node.js 使用 ...

  6. Shiro【授权、整合Spirng、Shiro过滤器】

    前言 本文主要讲解的知识点有以下: Shiro授权的方式简单介绍 与Spring整合 初始Shiro过滤器 一.Shiro授权 上一篇我们已经讲解了Shiro的认证相关的知识了,现在我们来弄Shiro ...

  7. 教你如何前后端完全分离(非api、ajax)

    我的前后分离,不是api,不是ajax,我这里只讨论html与后端结合 前话 曾经风靡一时的dedecms相信做网站的十有八.九都知道,还有那么一些不是技术出生的人,通过看一下文档,也能访问出网站出来 ...

  8. bat脚本:windows下一键启动zookeeper+kafka

    bat脚本:windows下一键启动zookeeper+kafka 把下面两行代码存为bat文件,双击执行即可.注意更改相应的目录 这里用ping来控制时间(先zookeeper,ping 4 次后 ...

  9. IntelliJ IDEA 2017.1.5迁移eclipse,SSM项目,通过jrebel实现热部署

    1.首先打开idea,配置SVN版本控制器的路径 2.配置maven 3.配置jrebel热部署的路径 4.从svn到出项目 5.配置配置tomacat参数-server -XX:PermSize=1 ...

  10. nodejs加密Crypto简单例子

    加密技术通常分为两大类:“对称式”和“非对称式”. 对称式加密: 就是加密和解密使用同一个密钥,通常称之为“Session Key ”这种加密技术在当今被广泛采用,如美国政府所采用的DES加密标准就是 ...