码迷,mamicode.com
首页 > 其他好文 > 详细

[bzoj3688]] 折线统计

时间:2018-12-02 16:18:28      阅读:136      评论:0      收藏:0      [点我收藏+]

标签:线段   geo   坐标   方案总数   ++   har   n+1   amp   void   

Description

二维平面上有n个点(xi, yi),现在这些点中取若干点构成一个集合S,对它们按照x坐标排序,顺次连接,将会构成一些连续上升、下降的折线,设其数量为f(S)。如下图中,1->2,2->3,3->5,5->6(数字为下图中从左到右的点编号),将折线分为了4部分,每部分连续上升、下降。
技术分享图片
现给定k,求满足f(S) = k的S集合个数。

Input

第一行两个整数n和k,以下n行每行两个数(xi, yi)表示第i个点的坐标。所有点的坐标值都在[1, 100000]内,且不存在两个点,x坐标值相等或y坐标值相等

Output

输出满足要求的方案总数 mod 100007的结果

Sample Input

5 1
5 5
3 2
4 4
2 3
1 1

Sample Output

19 

HINT

对于100%的数据,n <= 50000,0 < k <= 10

Source

题解

线段树/bit优化dp。

\(f[i][j][0/1]\)表示考虑了前\(i\)个点,有\(j\)段折线,且最后一段是上升/下降的方案数。

可以得到一个很显然的转移:
\[ f[i][k][0]=\sum_{j=1}^{i-1} f[j][k][0]*[a[i]>a[j]]+f[j][k-1][1]*[a[i]>a[j]] \]

\[ f[i][k][1]=\sum_{j=1}^{i-1} f[j][k-1][0]*[a[i]<a[j]]+f[j][k][1]*[a[i]<a[j]] \]

注意到\(k\)的范围只有10,可以暴力枚举后两维,然后用一颗权值线段树来维护第一维。

#include<bits/stdc++.h>
using namespace std;
void read(int &x){
    x=0;int f=1;char ch=getchar();
    for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-f;
    for(;isdigit(ch);ch=getchar())x=(x<<1)+(x<<3)+ch-'0';x*=f;
}
#define write(x) printf("%d\n",x)
#define maxn 100005
const int mod = 100007;
int f[maxn][11][2],n,K;
struct point {int x,y;}a[maxn];
inline int cmp(point x,point y) {return x.x<y.x;}
#define ls p<<1
#define rs p<<1|1
#define mid ((l+r)>>1)
struct segment_tree {
    int sum[maxn<<2];
    void update(int p) {sum[p]=sum[ls]+sum[rs];}
    void modify(int p,int l,int r,int x,int v) {
        if(l==r) return sum[p]=v%mod,void();
        if(x<=mid) modify(ls,l,mid,x,v);
        else modify(rs,mid+1,r,x,v);
        update(p);
    }
    int query(int p,int l,int r,int x,int y) {
        if(!y) return 0;
        if(x<=l&&r<=y) return sum[p];
        int ans=0;
        if(x<=mid) ans=(ans+query(ls,l,mid,x,y))%mod;
        if(y>mid) ans=(ans+query(rs,mid+1,r,x,y))%mod;
        update(p);return ans;
    }
}T[11][2];
#define N 100000
int main() {
    read(n),read(K);;for(int i=1;i<=n;i++) read(a[i].x),read(a[i].y);
    sort(a+1,a+n+1,cmp);
    //for(int i=1;i<=n;i++) printf("%d %d\n",a[i].x,a[i].y);
    for(int i=1;i<=n;i++) {
        f[i][0][0]=f[i][0][1]=1;
        T[0][0].modify(1,1,N,a[i].y,1),T[0][1].modify(1,1,N,a[i].y,1);
        for(int k=1;k<=K;k++) {
            f[i][k][0]=(T[k][0].query(1,1,N,1,a[i].y-1)+T[k-1][1].query(1,1,N,1,a[i].y-1))%mod;
            f[i][k][1]=(T[k][1].query(1,1,N,a[i].y+1,N)+T[k-1][0].query(1,1,N,a[i].y+1,N))%mod;
            T[k][0].modify(1,1,N,a[i].y,f[i][k][0]),T[k][1].modify(1,1,N,a[i].y,f[i][k][1]);
            //if(i==3) printf("::%d\n",T[k][1].query(1,1,N,5,5));
            //printf("%d %d\n",f[i][1][0],f[i][1][1]);
        }
    }int ans=0;
    //for(int i=1;i<=n;i++) printf("%d %d\n",f[i][1][0],f[i][1][1]);
    for(int i=1;i<=n;i++) ans=(ans+f[i][K][0]+f[i][K][1])%mod;
    write(ans);
    return 0;
}

[bzoj3688]] 折线统计

标签:线段   geo   坐标   方案总数   ++   har   n+1   amp   void   

原文地址:https://www.cnblogs.com/hbyer/p/10053650.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!