Berlekamp–Massey 算法
Berlekamp–Massey 算法是一种用于求数列的最短递推式的算法。给定一个长为
定义
定义一个数列
其中
数列
做法
与上面定义的稍有不同,这里定义一个新的递推系数
容易看出
我们可以增量地求递推式,按顺序考虑
显然初始时有
- 递推系数对
也成立,这时不需要进行任何调整,直接令 即可。 - 递推系数对
不成立,这时需要对 进行调整,得到新的 。
设
如果这是第一次对递推系数进行修改,则说明
否则设上一次对递推系数进行修改时,已考虑的
并且
考虑如何构造
其中前面一共有
不难验证此时
如果要求的是符合最开始定义的递推式
从上述算法流程中可以看出,如果数列的最短递推式的阶数为
在实现算法时,由于每次调整递推系数时都只需要用到上次调整时的递推系数
参考实现
vector<int> berlekamp_massey(const vector<int> &a) {
vector<int> v, last; // v is the answer, 0-based, p is the module
int k = -1, delta = 0;
for (int i = 0; i < (int)a.size(); i++) {
int tmp = 0;
for (int j = 0; j < (int)v.size(); j++)
tmp = (tmp + (long long)a[i - j - 1] * v[j]) % p;
if (a[i] == tmp) continue;
if (k < 0) {
k = i;
delta = (a[i] - tmp + p) % p;
v = vector<int>(i + 1);
continue;
}
vector<int> u = v;
int val = (long long)(a[i] - tmp + p) * power(delta, p - 2) % p;
if (v.size() < last.size() + i - k) v.resize(last.size() + i - k);
(v[i - k - 1] += val) %= p;
for (int j = 0; j < (int)last.size(); j++) {
v[i - k + j] = (v[i - k + j] - (long long)val * last[j]) % p;
if (v[i - k + j] < 0) v[i - k + j] += p;
}
if ((int)u.size() - i < (int)last.size() - k) {
last = u;
k = i;
delta = a[i] - tmp;
if (delta < 0) delta += p;
}
}
for (auto &x : v) x = (p - x) % p;
v.insert(v.begin(), 1);
return v; // $\forall i, \sum_{j = 0} ^ m a_{i - j} v_j = 0$
}
朴素的 Berlekamp–Massey 算法求解的是有限项数列的最短递推式。如果待求递推式的序列有无限项,但已知最短递推式的阶数上界,则只需取出序列的前
应用
由于 Berlekamp–Massey 算法的数值稳定性比较差,在处理实数问题时一般很少使用。为了叙述方便,以下均假定在某个质数
求向量列或矩阵列的最短递推式
如果要求向量列
求矩阵列
优化矩阵快速幂
设
我们可以直接暴力求出
如果要求的向量是
求矩阵的最小多项式
方阵
实际上最小多项式就是
瓶颈在于求出
假设
求稀疏矩阵行列式
如果能求出方阵
实际上如果把
设
求稀疏矩阵的秩
设
实际上不用调用矩阵乘法,因为求最小多项式时要用
设
解稀疏方程组
问题:已知
做法:显然
(证明略)
因为
同样地,设
参考实现
vector<int> solve_sparse_equations(const vector<tuple<int, int, int>> &A,
const vector<int> &b) {
int n = (int)b.size(); // 0-based
vector<vector<int>> f({b});
for (int i = 1; i < 2 * n; i++) {
vector<int> v(n);
auto &u = f.back();
for (auto [x, y, z] : A) // [x, y, value]
v[x] = (v[x] + (long long)u[y] * z) % p;
f.push_back(v);
}
vector<int> w(n);
mt19937 gen;
for (auto &x : w) x = uniform_int_distribution<int>(1, p - 1)(gen);
vector<int> a(2 * n);
for (int i = 0; i < 2 * n; i++)
for (int j = 0; j < n; j++) a[i] = (a[i] + (long long)f[i][j] * w[j]) % p;
auto c = berlekamp_massey(a);
int m = (int)c.size();
vector<int> ans(n);
for (int i = 0; i < m - 1; i++)
for (int j = 0; j < n; j++)
ans[j] = (ans[j] + (long long)c[m - 2 - i] * f[i][j]) % p;
int inv = power(p - c[m - 1], p - 2);
for (int i = 0; i < n; i++) ans[i] = (long long)ans[i] * inv % p;
return ans;
}
例题
- LibreOJ #163. 高斯消元 2
- ICPC2021 台北 Gym103443E. Composition with Large Red Plane, Yellow, Black, Gray, and Blue
创建日期: 2022年3月18日