Berlekamp-Massey 算法用于求解常系数线性递推式

#include<bits/stdc++.h>

typedef std::vector<int> VI;
typedef long long ll;
typedef std::pair<int, int> PII; const ll mod = 1000000007; ll powmod(ll a, ll b) {
ll res = 1;
a %= mod;
assert(b >= 0);
for(; b; b >>= 1) {
if(b & 1)
res = res * a % mod;
a = a * a % mod;
}
return res;
} ll _, n; namespace linear_seq {
const int N = 10010;
ll res[N], base[N], _c[N], _md[N];
std::vector<ll> Md;
void mul(ll *a, ll *b, int k) {
for (int i = 0; i < k + k; i++)
_c[i] = 0;
for (int i = 0; i < k; i++)
if (a[i])
for (int j = 0; j < k; j++)
_c[i + j] = (_c[i + j] + a[i] * b[j]) % mod;
for (int i = k + k - 1; i >= k; i--)
if (_c[i])
for (int j = 0; j < ((int)(Md).size()); j++)
_c[i - k + Md[j]] = (_c[i - k + Md[j]] - _c[i] * _md[Md[j]]) % mod;
for (int i = 0; i < k; i++)
a[i] = _c[i];
}
int solve(ll n, VI a, VI b) {
ll ans = 0, pnt = 0;
int k = ((int)(a).size());
assert(((int)(a).size()) == ((int)(b).size()));
for (int i = 0; i < k; i++)
_md[k - 1 - i] = -a[i];
_md[k] = 1;
Md.clear();
for (int i = 0; i < k; i++)
if (_md[i] != 0)
Md.push_back(i);
for (int i = 0; i < k; i++)
res[i] = base[i] = 0;
res[0] = 1;
while ((1ll << pnt) <= n)
pnt++;
for (int p = pnt; p >= 0; p--) {
mul(res, res, k);
if ((n >> p) & 1) {
for (int i = k - 1; i >= 0; i--)
res[i + 1] = res[i];
res[0] = 0;
for (int j = 0; j < ((int)(Md).size()); j++)
res[Md[j]] = (res[Md[j]] - res[k] * _md[Md[j]]) % mod;
}
}
for (int i = 0; i < k; i++)
ans = (ans + res[i] * b[i]) % mod;
if (ans < 0)
ans += mod;
return ans;
}
VI BM(VI s) {
VI C(1, 1), B(1, 1);
int L = 0, m = 1, b = 1;
for (int n = 0; n < ((int)(s).size()); n++) {
ll d = 0;
for (int i = 0; i < L + 1; i++)
d = (d + (ll)C[i] * s[n - i]) % mod;
if (d == 0)
++m;
else if (2 * L <= n) {
VI T = C;
ll c = mod - d * powmod(b, mod - 2) % mod;
while (((int)(C).size()) < ((int)(B).size()) + m)
C.push_back(0);
for (int i = 0; i < ((int)(B).size()); i++)
C[i + m] = (C[i + m] + c * B[i]) % mod;
L = n + 1 - L;
B = T;
b = d;
m = 1;
} else {
ll c = mod - d * powmod(b, mod - 2) % mod;
while (((int)(C).size()) < ((int)(B).size()) + m)
C.push_back(0);
for (int i = 0; i < ((int)(B).size()); i++)
C[i + m] = (C[i + m] + c * B[i]) % mod;
++m;
}
}
return C;
}
int gao(VI a, ll n) {
VI c = BM(a);
c.erase(c.begin());
for (int i = 0; i < ((int)(c).size()); i++)
c[i] = (mod - c[i]) % mod;
return solve(n, c, VI(a.begin(), a.begin() + ((int)(c).size())));
}
};
int main() {
int t;
scanf("%d", &t);
while(t--) {
scanf("%lld", &n);
std::vector<int>v = {
1, 1, 0, 3, 0, 3,
5, 4, 1, 9, 1, 6,
9, 7, 2, 15, 2, 9,
13, 10, 3, 21, 3, 12
};
//至少8项,越多越好。
printf("%lld\n", linear_seq::gao(v, n - 1) % mod);
}
}

数据大时都改为 long long

若mod不为质数,则需替换 powmod ,并将BM中的 d*powmod(b, mod-2) 改为 powmod(d, b)


void exgcd(ll a, ll b, ll &x, ll &y) {
if (!b)
x = 1, y = 0;
else
exgcd(b, a % b, y, x), y -= x * (a / b);
}
ll inv(ll a, ll p) {
ll x, y;
exgcd(a, p, x, y);
return(x + p) % p;
}
VI prime, g;
void getPrime() {
ll qw = mod;
for(ll i = 2; i * i <= qw; i++) {
if(qw % i == 0)
prime.pb(i);
while(qw % i == 0)
qw /= i;
}
if(qw > 1)
prime.push_back(qw);
}
ll powmod(ll fz, ll fm) {
ll ret = 1ll;
ll cnt[5] = {0};
for(int k = 0; k < prime.size(); k++) {
if(fz % prime[k] == 0) {
while(fz % prime[k] == 0) {
fz /= prime[k];
cnt[k]++;
}
}
}
ret = fz % mod;
for(int k = 0; k < prime.size(); k++) {
if(fm % prime[k] == 0) {
while(fm % prime[k] == 0) {
fm /= prime[k];
cnt[k]--;
}
}
}
if(fm > 1)
ret = (ret * inv(fm, mod)) % mod;
for(int k = 0; k < prime.size(); k++)
for(int kk = 1; kk <= cnt[k]; kk++)
ret = (ret * prime[k]) % mod;
return ret;
}

杜教BM递推板子的更多相关文章

  1. ZZNU 2182 矩阵dp (矩阵快速幂+递推式 || 杜教BM)

    题目链接:http://47.93.249.116/problem.php?id=2182 题目描述 河神喜欢吃零食,有三种最喜欢的零食,鱼干,猪肉脯,巧克力.他每小时会选择一种吃一包. 不幸的是,医 ...

  2. 牛客多校第九场 A The power of Fibonacci 杜教bm解线性递推

    题意:计算斐波那契数列前n项和的m次方模1e9 题解: $F[i] – F[i-1] – F[i-2] = 0$ $F[i]^2 – 2 F[i-1]^2 – 2 F[i-2]^2 + F[i-3] ...

  3. BM递推杜教版

    #include <bits/stdc++.h> using namespace std; #define rep(i,a,n) for (long long i=a;i<n;i++ ...

  4. 黑科技之杜教bm

    这个板子能够解决任何线性递推式,只要你确定某个数列的某项只与前几项线性相关,那么把它前若干项丢进去,这个板子就能给你返回你要求的某项的值. 原理???(待补充) #include<bits/st ...

  5. 2019牛客多校第二场BEddy Walker 2——BM递推

    题意 从数字 $0$ 除法,每次向前走 $i$ 步,$i$ 是 $1 \sim K$ 中等概率随机的一个数,也就是说概率都是 $\frac{1}{K}$.求落在过数字 $N$ 额概率,$N=-1$ 表 ...

  6. BM递推杜教版【扩展】

    也就是模数不是质数的时候, //下面的板子能求质数和非质数,只需要传不同的参数. #include <cstdio> #include <cstdlib> #include & ...

  7. BM递推

    从别的大佬处看到的模板 #include<bits/stdc++.h> #define fi first #define se second #define INF 0x3f3f3f3f ...

  8. 杜教BM【转载】

    https://blog.csdn.net/qq_36876305/article/details/80275708 #include <bits/stdc++.h> using name ...

  9. 杜教BM

    #include <algorithm> #include <iterator> #include <iostream> #include <cstring& ...

随机推荐

  1. [Luogu] 让我们异或吧

    https://www.luogu.org/problemnew/show/P2420 异或满足 A ^ B = B ^ A A ^ A = 0 0 ^ A = A #include <cstd ...

  2. 【luogu4145】上帝造题的七分钟2 / 花神游历各国--区间开根-线段树

    题目背景 XLk觉得<上帝造题的七分钟>不太过瘾,于是有了第二部. 题目描述 "第一分钟,X说,要有数列,于是便给定了一个正整数数列. 第二分钟,L说,要能修改,于是便有了对一段 ...

  3. linux中fork--子进程是从哪里开始运行

    转自  http://blog.csdn.net/koches/article/details/7787468 fork调用的一个奇妙之处就是它仅仅被调用一次,却能够返回两次,它可能有三种不同的返回值 ...

  4. Try-Catch-Finally代码块中的return

    测试类的原型是这样子的 public class TryCatchFinallyToReturn { public static void main(String[] args) { System.o ...

  5. pytest学习笔记(一)

    这两天在学习pytest,之前有小用到pytest,觉得这个测试框架很灵巧,用在实现接口自动化(pytest+requests)非常的轻便,然后很有兴致的决定学习下,然后又发现了pytest-sele ...

  6. spring事务配置异常

    spring事务配置不回滚spring事务管理配置,一般来说都是可以回滚的,最近在开发的过程中遇到了一个异常不回滚的问题,最终找到了原因,贴出来一下 1.首先这里定义一个接口 在接口中定义几个方法 2 ...

  7. ICEM—气体罐子

    原视频下载地址:https://yunpan.cn/cPfzaLzkcFXVK  访问密码 62cf

  8. pwn学习日记Day10 《程序员自我修养》读书笔记

    第一章 从 Hello world 说起 抛出问题: 1.程序为什么要被编译器编译后才能执行? 2.编译器在把C语言程序转换成可以执行的机器码的过程中做了什么,怎么做的? 3.最后编译出来的可执行文件 ...

  9. Qt configure 参数

    在编译QT前,可加各种参数来定制自己想要的QT库.这对需要裁减QT库的朋友来说非常重要.对于如何编译QT,可以参考:http://hi.baidu.com/agassi%5Fp/blog/item/4 ...

  10. JS -- Unexpected trailing comma

    Unexpected trailing comma 后面多了一个逗号