杜教BM递推板子
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递推板子的更多相关文章
- ZZNU 2182 矩阵dp (矩阵快速幂+递推式 || 杜教BM)
题目链接:http://47.93.249.116/problem.php?id=2182 题目描述 河神喜欢吃零食,有三种最喜欢的零食,鱼干,猪肉脯,巧克力.他每小时会选择一种吃一包. 不幸的是,医 ...
- 牛客多校第九场 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] ...
- BM递推杜教版
#include <bits/stdc++.h> using namespace std; #define rep(i,a,n) for (long long i=a;i<n;i++ ...
- 黑科技之杜教bm
这个板子能够解决任何线性递推式,只要你确定某个数列的某项只与前几项线性相关,那么把它前若干项丢进去,这个板子就能给你返回你要求的某项的值. 原理???(待补充) #include<bits/st ...
- 2019牛客多校第二场BEddy Walker 2——BM递推
题意 从数字 $0$ 除法,每次向前走 $i$ 步,$i$ 是 $1 \sim K$ 中等概率随机的一个数,也就是说概率都是 $\frac{1}{K}$.求落在过数字 $N$ 额概率,$N=-1$ 表 ...
- BM递推杜教版【扩展】
也就是模数不是质数的时候, //下面的板子能求质数和非质数,只需要传不同的参数. #include <cstdio> #include <cstdlib> #include & ...
- BM递推
从别的大佬处看到的模板 #include<bits/stdc++.h> #define fi first #define se second #define INF 0x3f3f3f3f ...
- 杜教BM【转载】
https://blog.csdn.net/qq_36876305/article/details/80275708 #include <bits/stdc++.h> using name ...
- 杜教BM
#include <algorithm> #include <iterator> #include <iostream> #include <cstring& ...
随机推荐
- 【线性代数】5-1:行列式性质(The Properties of Determinants)
title: [线性代数]5-1:行列式性质(The Properties of Determinants) categories: Mathematic Linear Algebra keyword ...
- LOJ166 拉格朗日插值 2【卷积,NTT】
题目链接:LOJ 题目描述:输入多项式的次数$n$,一个整数$m$和$f(0),f(1),f(2),\ldots,f(n)$,输出$f(m),f(m+1),f(m+2),\ldots,f(m+n)$ ...
- [python]字典的直接赋值、浅拷贝和深拷贝解析
1.赋值引用 b = a: a 和 b 都指向同一个对象. 2.浅拷贝 b = a.copy(): a 和 b父对象是一个独立的对象,但他们的子对象还是指向统一对象(是引用). 3.深拷贝 b = ...
- deepin安装下载,部署在虚拟机上
第一次接触linux就用了很久的Ubuntu,后来尝试了Manjaro.Debian,还是用了Ubuntu,但是用VM虚拟机使用的ubuntu经常卡顿.我始终觉得ubuntu的默认桌面环境gnome可 ...
- Ocelot 网关 和 consul 服务发现
服务发现 Consul 一.安装和启动 下载 [Consul](https://www.consul.io/downloads.html) 下载完成后,解压,只有一个consul.exe,把目录添加到 ...
- 在Vue文件中引用模块的相对路径“@“符号表示什么意思?
@ 的作用是在你引入模块时,可以使用 @ 代替 /src 目录,避免书写麻烦又易错的相对路径. import model from "@/common/model"; // 默认路 ...
- MongoDB作为windows服务来安装-2
首先区官网下载对应版本的安装文件,我本地的环境是win7 bit64 我下载的版本是:mongodb-win32-x86_64-2.4.6 ok, 文件下载后,开始安装,这里要说一下,如果直接启动Mo ...
- ME21N屏幕格式配置路径
物料管理->采购->采购订单->定义屏幕层的屏幕格式
- 以rpm安装包的方式安装MySQL
rpm -vif MySQL-server-5.6.26-1.linux_glibc2.5.x86_64.rpm MySQL-client-5.6.26-1.linux_glibc2.5.x86_64 ...
- Sprint Retrospective - 回顾的重要性
在Scrum中,每个Sprint结束的时候会有两个会议(Sprint Review/Demo和Sprint Retrospective回顾).这两个会议是对过去的一个Sprint的一个总结,其中Rev ...