hdu 4869

维护最大能翻多少张,最少能翻多少张,然后从最大到最小,奇偶相同的也可以翻到。

维护,然后求组合数,优化是求组合数的时候如果i > m - i的话,那么c[i] = c[m - i],否则会TLE,常数优化,被坑了好久。

#include <cstdio>
#include <cstring>
#include <iostream>

using namespace std;
const long long mod = 1000000009ll;
const int maxn = 1000007;
//const int mod = 1000 * 1000 * 1000 + 9;

long long c[maxn];

long long powMod(long long a, long long b) {
    long long result = 1;
    long long cnt = a;
    while (b) {
        if (b & 1) {
            result = (result * cnt) % mod;
        }
        cnt = (cnt * cnt) % mod;
        b /= 2;
    }
    return result % mod;
}

int n, m;
int main() {
//    freopen("in.txt", "r", stdin);
    while (scanf("%d%d", &n, &m) != EOF) {
        int l = 0, r = 0;
        for (int i = 1; i <= n; i++) {
            int x;
            scanf("%d", &x);
            int next_l = l;
            int next_r = r;
            if (l >= x) {
                next_l = l - x;
            }
            else if (r >= x) {
                next_l = ((x & 1) == (r & 1)) ? 0 : 1;
            }
            else {
                next_l = x - r;
            }
            if (x + r <= m) {
                next_r = x + r;
            }
            else if (l + x <= m) {
                next_r = ((m - l) & 1) ==  (x & 1) ? m : m - 1;
            }
            else {
                next_r = m - (x - (m - l));
            }
            l = next_l;
            r = next_r;
        }
        c[0] = 1;
        long long ans = 0;
        for (int i = 0; i <= r; i++) {
            if (i) {
                if (m - i < i) {
                    c[i] = c[m - i];
                }
                else {
                    c[i] = c[i - 1] * ((long long)(m - i + 1) * powMod(i, mod - 2) % mod) % mod;
                }
            }
            if (i >= l && (i & 1) == (l & 1)) {
                ans = (ans + c[i]) % mod;
            }
        }
        printf("%I64d\n", ans);
    }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值