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. 【BZOJ1501】【NOI2005】智慧珠游戏(搜索)

    [BZOJ1501][NOI2005]智慧珠游戏(搜索) 题面 我要一改我懒惰的作风 这道题目必须放题面 Description Input 文件中包含初始的盘件描述,一共有10行,第i行有i个字符. ...

  2. 【BZOJ3730】震波(动态点分治)

    [BZOJ3730]震波(动态点分治) 题面 BZOJ 题意 给定一棵树, 每次询问到一个点的距离\(<=K\)的点的权值之和 动态修改权值, 强制在线 题解 正常的\(DP\)??? 很简单呀 ...

  3. XAMPP搭建PHP

    在学习一些前后端交互时,经常会有跟PHP作为后端(服务器)的交互,不能将php文件放在本地进行请求,必须将PHP运行在Apache环境中.但是对一些新手来说,学习搭建一个Apache环境也并非易事,所 ...

  4. imageview无法显示图片:java.lang.RuntimeException: Canvas: trying to draw too large(281520000bytes) bitmap

    图片太大需要压缩. 压缩方法:http://jingyan.baidu.com/article/cdddd41c3ef41153ca00e162.html 如果特别大(几十M),可以先用在线的图片压缩 ...

  5. Tencent研发工程师笔试知识点

      一: 32位编译器:32位系统下指针占用4字节       char :1个字节       char*(即指针变量): 4个字节(32位的寻址空间是2^32, 即32个bit,也就是4个字节.同 ...

  6. 13.C++-静态成员变量、静态成员函数

    首先回顾下成员变量 能通过对象名能够访问public成员变量 每个对象的成员变量都是专属的 成员变量不能在对象之间共享 再来讲讲类的静态成员变量 介绍 静态成员变量属于整个类所有 静态成员变量的生命期 ...

  7. 关于js高度和宽度的获取 ----2017-03-29

    来源:百度  对错有待实践检验 网页可见区域宽: document.body.clientWidth 网页可见区域高: document.body.clientHeight 网页可见区域宽: docu ...

  8. PHP自动测试框架Top 10

    对于很多PHP开发新手来说,测试自己编写的代码是一个非常棘手的问题.如果出现问题,他们将不知道下一步该怎么做.花费很长的时间调试PHP代码是一个非常不明智的选择,最好的方法就是在编写应用程序代码之前就 ...

  9. ArrayList源码解析(JDK1.8)

    package java.util; import sun.misc.SharedSecrets; import java.util.function.Consumer; import java.ut ...

  10. shiro授权

    一.shiro-permission.ini shiro-permission.ini里面的内容相当于在数据库 #用户 [users] #用户zhang的密码是123,此用户具有role1和role2 ...