杜教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& ...
随机推荐
- 一些特殊的矩阵快速幂 hdu5950 hdu3369 hdu 3483
思想启发来自, 罗博士的根据递推公式构造系数矩阵用于快速幂 对于矩阵乘法和矩阵快速幂就不多重复了,网上很多博客都有讲解.主要来学习一下系数矩阵的构造 一开始,最一般的矩阵快速幂,要斐波那契数列Fn=F ...
- CF540D Bad Luck Island
嘟嘟嘟 看到数据范围很小,就可以暴力\(O(n ^ 3)\)dp啦. 我们令\(dp[i][j][k]\)表示这三种人分别剩\(i, j, k\)个的概率.然后枚举谁挂了就行. 这里的重点在于两个人相 ...
- Django—Ajax
Ajax-get url url(r'^ajax_add/', views.ajax_add), url(r'^ajax_demo1/', views.ajax_demo1), 视图 def ajax ...
- 数据结构实验之栈与队列二:一般算术表达式转换成后缀式(SDUT 2132)
题目链接 #include <bits/stdc++.h> using namespace std; typedef long long ll; int ok(char ch, char ...
- java代理,手把手交你写java代理
一:常用的java代理模式 一般经常做java开发的知道java的代理模式一共有三种,第一种也就是静态代理,这种用法比较简单,没有什么魔法棒,比较好理解,另外两种分别是JDK代理和cglib代理,他们 ...
- Oracle impdp导入数据报错:无法读取要读取的存储文件(Linux)
当向Linux下的Oracle11g通过数据泵impdp导入数据库时,出现如图所示错误. 错误原因:bdck.dmp该为大写. 切记:Linux系统严格区分大小写.
- ios TCP抓包
1.工具安装 Mac15.1 ,Xcode11.2.1 这一步有个小坑,我第一次执行rvictls -s 提示 rvictl: command not found.Stack Overflow上提供了 ...
- Default Keyboard Shortcut Schemes
Default Keyboard Shortcut Schemes All ReSharper actions can be invoked with keyboard shortcuts. Most ...
- C11中的Unicode
在C11(ISO/IEC 9899:2011)标准中引入了对UTF8.UTF16以及UTF32字符编码的支持. 其中,UTF8字符直接通过char来定义,字面量前缀使用u8.比如: char c = ...
- 启动elasticsearch的时候报出Exception in thread "main" SettingsException[Failed to load settings from /usr/local/elasticsearch/config/elasticsearch.yml]; nested: MarkedYAMLException[while scanning a simple ke
故障现象: [elasticsearch@tiantianml- ~]$ /usr/local/elasticsearch/bin/elasticsearch Exception in thread ...