• 洛谷2344 奶牛抗议(DP+BIT+离散化)


    洛谷2344 奶牛抗议

    本题地址: http://www.luogu.org/problem/show?pid=2344

    题目背景

    Generic Cow Protests, 2011 Feb

    题目描述

    约翰家的N 头奶牛正在排队游行抗议。一些奶牛情绪激动,约翰测算下来,排在第i 位的奶牛的理智度为Ai,数字可正可负。
    约翰希望奶牛在抗议时保持理性,为此,他打算将这条队伍分割成几个小组,每个抗议小组的理智度之和必须大于或等于零。奶牛的队伍已经固定了前后顺序,所以不能交换它们的位置,所以分在一个小组里的奶牛必须是连续位置的。除此之外,分组多少组,每组分多少奶牛,都没有限制。
    约翰想知道有多少种分组的方案,由于答案可能很大,只要输出答案除以1000000009 的余数即可。

    输入输出格式

    输入格式:

    • 第一行:单个整数N,1 ≤ N ≤ 100000
    • 第二行到第N + 1 行:第i + 1 行有一个整数Ai,−10^5 ≤ Ai ≤ 10^5

    输出格式:

    单个整数:表示分组方案数模1000000009 的余数

    输入输出样例

    输入样例#1:

    4
    2
    3
    -3
    1

    输出样例#1:

    4

    说明

    解释:如果分两组,可以把前三头分在一组,或把后三头分在一组;如果分三组,可以把中间两头分在一组,第一和最后一头奶牛自成一组;最后一种分法是把四头奶牛分在同一组里。

    【思路】

      线性DP+BIT加速+离散化。

      设d[i]表示前i头奶牛的分法,则有转移方程式如下:

         d[i]=sigma{ d[j]( j<i && S(j+1,i)>=0 ) }

      其中sigma表示求和、S代表区间和。

      如果令sum[]表示前缀和,则可以进一步得出转移条件:存在j<i且sum[j]<=sum[i]

      BIT加速:如果DP枚举到i,令C[x]表示i之前sum==x的所有d之和,则d[i]为小于sum[i]的所有d之和,可以用BIT求出小于sum[i]的区间和。

      离散化:sum的情况最多有n+1种而其范围可能很大,所以考虑对sum进行离散化。

      另外有0的情况可以考虑将BIT下标进行偏移或hash到其他范围。

    【代码】

     1 #include<cstdio>
     2 #include<iostream>
     3 #include<algorithm>
     4 #define FOR(a,b,c) for(int a=(b);a<=(c);a++)
     5 using namespace std;
     6 
     7 const int maxn = 100000+10;
     8 const int MOD=1000000009;
     9 
    10 int sum[maxn],a[maxn];
    11 int hash[maxn],cnt;
    12 int n;
    13 
    14 int C[maxn],Max;
    15 int lowbit(int x) { return x&(-x); }
    16 int Sum(int x) {
    17     x++;
    18     int res=0;
    19     while(x>0) {
    20         res = (res+C[x])%MOD;
    21         x-=lowbit(x);
    22     }
    23     return res;
    24 }
    25 void Add(int x,int v) {
    26     x++;
    27     while(x<=Max+1) {
    28         C[x] = (C[x]+v)%MOD;
    29         x+=lowbit(x);
    30     }
    31 }
    32 
    33 int find(int x) {
    34     return lower_bound(hash,hash+cnt+1,x)-hash;
    35 }
    36 
    37 int main() {
    38     scanf("%d",&n);
    39     FOR(i,1,n) {
    40         scanf("%d",&a[i]);
    41         sum[i]=sum[i-1]+a[i];
    42         Max=max(Max,sum[i]);
    43     }
    44     sort(sum,sum+n+1);         //将0计入 
    45     hash[0]=sum[0];
    46     FOR(i,1,n) if(sum[i]!=sum[i-1]) {
    47         hash[++cnt]=sum[i];
    48     }
    49     Add(find(0),1);
    50     int tot=0,ans=0;
    51     FOR(i,1,n) {
    52         tot += a[i];
    53         ans = Sum(find(tot))%MOD;
    54         Add(find(tot),ans);
    55     }
    56     printf("%d
    ",ans);
    57     return 0;
    58 }
  • 相关阅读:
    P4016 负载平衡问题 网络流
    P3357 最长k可重线段集问题 网络流
    mysql部署
    pve配置
    PVE手册资料
    PVE授权条款
    ovirt磁盘类型(IDE, virtio, virtio-scsi)
    oVirt-postgresql
    ovirt常用后台维护命令
    oVirt部署
  • 原文地址:https://www.cnblogs.com/lidaxin/p/4924259.html
Copyright © 2020-2023  润新知