Berlekamp_Massey 算法 (BM算法) 學習筆記


原文鏈接www.cnblogs.com/zhouzhendong/p/Berlekamp-Massey.html

前言

BM算法用於求解常系數線性遞推式。

它可以在 $O(n^2)$ 的時間復雜度內解決問題。

由於許多問題會涉及線性遞推,所以 BM 算法將會有不錯的應用。

問題模型

給定一個有 $n$ 個元素的數列 $a$,其中第 $i$ 個元素是 $a_i$ 。

求一個 較短/最短 的數列 $b$,假設 $b$ 有 $m$ 個元素,那么要求滿足

$$\forall m<i\leq n, \ \ \ a_i = \sum_{j=1}^m a_{i-j} b_j$$

要求在 $O(n^2)$ 的時間復雜度內解決此問題。

BM算法

  考慮增量法。

  設遞推式經過了 $c$ 次更新,第 $i$ 次更新后的遞推式為 $R[i]$ 。初始時,定義 $R[0]$ 為空。

  考慮在當前數列末尾加入 $a_i$ 。假設當前遞推式長度為 $m$ 。

  設 $delta_i = a_i - \sum_{j=1}^m a_{i-j} R[c][j]$ 。

  如果 $delta_i = 0$ ,那么遞推式 $R[c]$ 依然合法,不用修改。

  否則,設 $Fail_{c} = i$ 表示遞推式 $R[c]$ 第一次失效的位置為 $i$ 。

  如果 $c = 0$ ,說明 $a_i$ 之前都是 0 ,顯然新的遞推式由 $i$ 個 $0$ 組成。

  考慮 $c\neq 0$ 的情況:考慮構造一個遞推式 $R'$ 使得對於 $|R'|+1\leq k < i$,$\sum_j a_{k-j} R'_j = 0$;$\sum_j a_{i-j} R'_j = delta_i$ 。

  設 $0\leq id < c$,設 $tmp = \frac{delta_i}{delta_{Fail[id]}}$,則我們考慮構造

$$R' = \{ 0,0,\cdots, 0,  tmp, -tmp\cdot R[id][1],- tmp \cdot R[id][2],\cdots \}$$

  其中開頭有 $i - Fail[id] - 1$ 個 0,$tmp$ 之后是 $-tmp$ 倍的 $R[id]$ 。

  容易證明,這個 $R'$ 符合要求。

  令 $R[c+1] = R[c] + R'$ 即可。

  至此,我們可以在 $O(n^2)$ 的時間復雜度內,求出數列 $a_i$ 的一個較短線性遞推式。

  那么如何求最短的線性遞推式呢?

  只要在對 $id$ 取值時,每次找 $i - Fail[id] + len(R[id])$ 最短的即可。

模板

#include <bits/stdc++.h>
#define clr(x) memset(x,0,sizeof (x))
#define For(i,a,b) for (int i=a;i<=b;i++)
#define Fod(i,b,a) for (int i=b;i>=a;i--)
#define pb(x) push_back(x)
#define mp(x,y) make_pair(x,y)
#define fi first
#define se second
#define _SEED_ ('C'+'L'+'Y'+'A'+'K'+'I'+'O'+'I')
#define outval(x) printf(#x" = %d\n",x)
#define outvec(x) printf("vec "#x" = ");for (auto _v : x)printf("%d ",_v);puts("")
#define outtag(x) puts("----------"#x"----------")
#define outarr(a,L,R) printf(#a"[%d...%d] = ",L,R);\
						For(_v2,L,R)printf("%d ",a[_v2]);puts("");
using namespace std;
typedef long long LL;
typedef unsigned long long ULL;
typedef vector <int> vi;
LL read(){
	LL x=0,f=0;
	char ch=getchar();
	while (!isdigit(ch))
		f|=ch=='-',ch=getchar();
	while (isdigit(ch))
		x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
	return f?-x:x;
}
const int N=0x1233,mod=1e9+7;
void Add(int &x,int y){
	if ((x+=y)>=mod)
		x-=mod;
}
void Del(int &x,int y){
	if ((x-=y)<0)
		x+=mod;
}
int Pow(int x,int y){
	int ans=1;
	for (;y;y>>=1,x=(LL)x*x%mod)
		if (y&1)
			ans=(LL)ans*x%mod;
	return ans;
}
int n,cnt;
int a[N];
int Fail[N],delta[N];
vector <int> R[N];
int main(){
	n=read();
	For(i,1,n)
		a[i]=read();
	R[0].clear();
	cnt=0;
	For(i,1,n){
		if (cnt==0){
			if (a[i]){
				Fail[cnt++]=i;
				delta[i]=a[i];
				R[cnt].resize(0);
				R[cnt].resize(i,0);
			}
			continue;
		}
		int sum=0,m=R[cnt].size();
		delta[i]=a[i];
		Fail[cnt]=i;
		For(j,0,m-1)
			Add(sum,(LL)a[i-j-1]*R[cnt][j]%mod);
		Del(delta[i],sum);
		if (!delta[i])
			continue;
		int id=cnt-1,v=i-Fail[id]+(int)R[id].size();
		For(j,0,cnt-1)
			if (i-Fail[j]+(int)R[j].size()<v)
				id=j,v=i-Fail[j]+(int)R[j].size();
		int tmp=(LL)delta[i]*Pow(delta[Fail[id]],mod-2)%mod;
		R[cnt+1]=R[cnt];
		while (R[cnt+1].size()<v)
			R[cnt+1].pb(0);
		Add(R[cnt+1][i-Fail[id]-1],tmp);
		For(j,0,(int)R[id].size()-1)
			Del(R[cnt+1][i-Fail[id]+j],(LL)tmp*R[id][j]%mod);
		cnt++;
	}
	printf("%d\n",(int)R[cnt].size());
	For(i,0,(int)R[cnt].size()-1)
		printf("%d ",R[cnt][i]);
	puts("");
	return 0;
}

關於模板的測試

到這篇博文寫完為止,各大OJ似乎並沒有BM算法的模板題。因此這里說明兩個測試數據來源:

1. cz_xuyixuan 的博客中的例子、評論中的數據:

Input 1
7
1 2 4 9 20 40 90

Output 1
4 
0 0 10 0

Input 2 
18
2 4 8 16 32 64 128 256 512 2 4 8 16 32 64 128 256 512

Output 2 
0 0 0 0 0 0 0 0 1

2. fjzzq2002 的博客中給出的數據。

相應博客鏈接見“參考文獻”。

參考文獻

https://blog.csdn.net/qq_39972971/article/details/80725873

http://www.cnblogs.com/zzqsblog/p/6877339.html

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM