LQ0214 波动数列DP

Posted 海岛Blog

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了LQ0214 波动数列DP相关的知识,希望对你有一定的参考价值。

题目来源:蓝桥杯2014初赛 C++ A组J题

题目描述
观察这个数列:

1 3 0 2 −1 1 −2 ⋯
这个数列中后一项总是比前一项增加 2 或者减少 3。

栋栋对这种数列很好奇,他想知道长度为 n 和为 s 而且后一项总是比前一项增加 a 或者减少 b 的整数数列可能有多少种呢?

输入描述
输入的第一行包含四个整数 n,s,a,b,含义如前面说述。

其中,1≤n≤1000,−109 ≤s≤109 ,1≤a,b≤106

输出描述
输出一行,包含一个整数,表示满足条件的方案数。由于这个数很大,请输出方案数除以 108 +7的余数。

输入输出样例
示例
输入

4 10 2 3

输出

2

样例说明

这两个数列分别是 2 4 1 3 和 7 4 1 -2。

问题分析
DP问题。

AC的C语言程序如下:

/* LQ0214 波动数列 */

#include <stdio.h>

typedef long long LL;

#define MOD 100000007
#define N 1000
LL dp[N + 1][N + 1];

int main()

    int n, s, a, b;
    scanf("%d%d%d%d", &n, &s, &a, &b);
    dp[1][(s % n + n) % n] = 1;
    for (int i = 2; i <= n; i++) 
        LL suma = (LL)a * (n - i + 1) % n;
        LL sumb = (LL)b * (n - i + 1) % n;
        for (int j = 0; j < n; j++) 
            dp[i][(j - suma + n) % n] = (dp[i][(j - suma + n) % n] + dp[i - 1][j]) % MOD;
            dp[i][(j + sumb) % n] = (dp[i][(j + sumb) % n] + dp[i - 1][j]) % MOD;
        
    
    printf("%lld\\n", dp[n][0]);

    return 0;

以上是关于LQ0214 波动数列DP的主要内容,如果未能解决你的问题,请参考以下文章

波动数列 dp

P2467 [SDOI2010]地精部落 DP

LQ0104 选数异或DP

bzoj2431[HAOI2009]逆序对数列 dp

LQ0115 外星日历余数定理

[bzoj2431][HAOI2009][逆序对数列] (dp计数)