Codeforces 193 D. Two Segments

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+1],b[N+1];

int tag[N<<2];
int tot[N<<2],tot1[N<<2];
int mi[N<<2];

long long ans=0;

void read(int &x)
{
    x=0; char c=getchar();
    while(!isdigit(c)) c=getchar();
    while(isdigit(c)) { x=x*10+c-'0'; c=getchar(); }
}

void build(int k,int l,int r)
{
    tot[k]=r-l+1;
    if(l==r) return;
    int mid=l+r>>1;
    build(k<<1,l,mid);
    build(k<<1|1,mid+1,r);
}

void down(int k)
{
    tag[k<<1]+=tag[k];
    tag[k<<1|1]+=tag[k];
    mi[k<<1]+=tag[k];
    mi[k<<1|1]+=tag[k];
    tag[k]=0;
}

void update(int k)
{
    mi[k]=min(mi[k<<1],mi[k<<1|1]);
    tot[k]=tot[k<<1]*(mi[k<<1]==mi[k])+tot[k<<1|1]*(mi[k<<1|1]==mi[k]);
    tot1[k]=tot1[k<<1]*(mi[k<<1]==mi[k])+tot1[k<<1|1]*(mi[k<<1|1]==mi[k]);
    tot1[k]+=tot[k<<1]*(mi[k<<1]==mi[k]+1)+tot[k<<1|1]*(mi[k<<1|1]==mi[k]+1);
}

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>>1;
    if(opl<=mid) change(k<<1,l,mid,opl,opr,w);
    if(opr>mid) change(k<<1|1,mid+1,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]<=2)+tot1[k]*(mi[k]==1);
        return;
    }
    if(tag[k]) down(k);
    int mid=l+r>>1;
    if(opl<=mid) query(k<<1,l,mid,opl,opr);
    if(opr>mid) query(k<<1|1,mid+1,r,opl,opr);
}    

int main()
{
    int n,m;
    read(n);
    int x,y;
    for(int i=1;i<=n;++i) read(x),a[x]=i;
    build(1,1,n);
    for(int i=n;i;--i)
    {
        b[a[i]]=i;
        x=b[a[i]-1];
        y=b[a[i]+1];
        if(x>y) swap(x,y);
        if(x)
        {
            change(1,1,n,y,n,-1);
            change(1,1,n,i,x-1,1);
        }
        else if(y) change(1,1,n,i,y-1,1);
        else change(1,1,n,i,n,1);
        //long long last=ans;
        query(1,1,n,i,n);
        //cout<<i<<' '<<ans-last<<'\n';
    }
    cout<<ans-n;
}

 

posted @ 2018-02-12 15:25  TRTTG  阅读(340)  评论(0编辑  收藏  举报