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. Gedit : 我的开场白 [TPLY]

    为什么用Gedit 在学校的高一新生里,好像就只有我使用Gedit 大家都笑我是"用记事本编程的人" 我就想 到考场看看你们笑得出来不 先放一个高配emacs配置 (global- ...

  2. 【BZOJ3277】串(后缀自动机)

    [BZOJ3277]串(后缀自动机) 题面 BZOJ 题解 广义后缀自动机??? 照着别人的打了一遍.. 相当于每个串都构建一个后缀自动机 构建完一个串之后,直接把当前的last指回root就行了?? ...

  3. 【NOIP2015】斗地主(搜索,贪心)

    题面戳我 题解 我原来也觉得是一道不可做的难题.. 其实,,,很简单的啦... 对于当前状态 我们出牌的方式大致分为两类 一类是不用考虑点数的,包括单张,对子,三带一等 另一类就是需要考虑点数的,包括 ...

  4. 一步一步创建ASP.NET MVC5程序[Repository+Autofac+Automapper+SqlSugar](十一)

    前言 小伙伴们, 大家好,我是Rector. 最近Rector忙于换工作,没有太多时间来更新我们的ASP.NET MVC 5系列文章 [一步一步创建ASP.NET MVC5程序Repository+A ...

  5. 链接生成二维码-PHP

    原文:http://www.upwqy.com/details/20.html 链接生成二维码 首先下载phpqrcode phpqrcode.zip 我这里使用的是TP5,把下载好的类库 放入到ex ...

  6. 深度剖析HashMap的数据存储实现原理(看完必懂篇)

    深度剖析HashMap的数据存储实现原理(看完必懂篇) 具体的原理分析可以参考一下两篇文章,有透彻的分析! 参考资料: 1. https://www.jianshu.com/p/17177c12f84 ...

  7. c#抽取pdf文档标题(3)

    上一篇介绍了整体流程以及利用库读取pdf内容形成字符集合.这篇着重介绍下,过滤规则,毕竟我们是使用规则过滤,最后得到标题的. 首先看归一化处理,什么是归一化呢?就是使结果始终处于0-1之间(包括0,1 ...

  8. 反射、Attribute

    1.发射是对类或者对象,查看其类内部的构造. 2.类的组成:属性(PropertyInfo).方法(MethodInfo).字段(FiedInfo).构造函数(ConstructorInfo).事件( ...

  9. VS快速注释

    注释:Ctrl+k  + Ctrl+c 去注释:Ctrl+k + Ctrl +u

  10. php做的一个简易爬虫

    对于爬虫的好奇好像由来已久,一直在研究python的爬虫,今天得空研究研究php的爬虫 index.php <?php header("Content-Type:text/html;c ...