Berlekamp_Massey 算法 (BM算法) 学习笔记

摘要:
由于许多问题涉及线性递归,BM算法将有很好的应用。查找较短/最短的序列$b$。假设$b$具有$m$元素,则$$forallm˂ileqn,a_i=sum_{j=1}^ma_{i-j}b_j$$要求在$O(n^2)$的时间复杂度内解决此问题。BM算法考虑增量方法。如果$delta_If i=0$,则递归$R[c]$仍然合法,不需要修改。考虑$ceq0$的情况:考虑构造递归的$R'$,使$|R'|+1eqk˂i$,$sum_ja_{k-j}R'_j=0$;$sum_ja_{i-j}R'_j=delta_i$delta[i])继续;intid=cnt-1,v=i-Fail[id]+R[id]。size();假设d=j,v=i-Fail[j]+R[j]。size();inttmp=delta[i]*Pow%mod;R[cnt+1]=R[cnt];而R[cnt+1]。pb;添加ForDel;cnt++;}输出函数Forprintf;看跌期权(“”);return0;}就这篇博文而言,对于主要的OJ来说,BM算法似乎没有模板问题。

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

前言

BM算法用于求解常系数线性递推式。

它可以在 $O(n^2)$ 的时间复杂度内解决问题。

由于许多问题会涉及线性递推,所以 BM 算法将会有不错的应用。

问题模型

给定一个有 $n$ 个元素的数列 $a$,其中第 $i$ 个元素是 $a_i$ 。

求一个 较短/最短 的数列 $b$,假设 $b$ 有 $m$ 个元素,那么要求满足

$$forall m<ileq 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 eq 0$ 的情况:考虑构造一个递推式 $R'$ 使得对于 $|R'|+1leq k < i$,$sum_j a_{k-j} R'_j = 0$;$sum_j a_{i-j} R'_j = delta_i$ 。

  设 $0leq id < c$,设 $tmp = frac{delta_i}{delta_{Fail[id]}}$,则我们考虑构造

$$R' = { 0,0,cdots, 0,  tmp, -tmpcdot 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
",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
",(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

免责声明:文章转载自《Berlekamp_Massey 算法 (BM算法) 学习笔记》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇表格td内容过多时,td显示省略号,鼠标移入显示全部内容。HTML5 创建热点图下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

随便看看

Centos7 挂载

1.Mount命令:Mount语法格式:Mount Mount设备文件信息Mount point(目录)注意:装载点(目录)必须有一个装载CD-ROM驱动器:Mount/dev/cdrom/mnt 2.卸载命令:umount语法格式:umountmount point(directory)3.查看磁盘装载状态/查看磁盘使用情况df4。存储设备通电时自动装载#...

java中cookie存取值

Cookie保存值:CookieuserCookie=newCookie(“loginInfo”,loginInfo);userCookie.setMaxAge(30*24*60*60);//生存期为一个月30*24*60*60userCookie.setPath(“/”);response.addCookie(userCookie);Cookie值:Coo...

如何在Android模拟器上安装apk文件

如本实例的“mishop_2.0.20130911_1.1.1.apk”3.执行控制台命令,进行安装。切换到D盘,输入D:,然后点击Enter,即切换到D盘,输入cd,找到platform-tools的文件地址,即adb.exe的文件路径。,粘贴在控制台中。...

应用程序-特定 权限设置并未向在应用程序容器 不可用 SID (不可用)中运行的地址 LocalHost (使用 LRPC) 中的用户 NT AUTHORITYSYSTEM SID (S-1-5-18)授予针对 CLSID 为 {D63B10C5-BB46-4990-A94F-E40B9D520

此安全权限可以使用组件服务管理工具进行修改。根据APPID为{9CA88EE3-ACB7-47C8-AFC4-AB702511C276}在注册表中找到HKEY_CLASSES_ROOTAppID{9CA88EE3-ACB7-47c8-AFC4-AB702511C276}右键选择权限:加入SYSTEM用户并赋予完全控制权限:如果在注册表中没有权限添加用户,则需...

myEclipse

因此,更改windows–>preferences–>general–>editors–>fileassociations,将myeclipsejspedator设置为默认的myeclipse使用提示步骤5:更改代码提示快捷键。当前代码提示快捷键默认为ctrl+space,我们的输入方法也被切换,因此会出现冲突。...

[RTOS]--uCOS、FreeRTOS、RTThread、RTX等RTOS的对比之特点

本博客将详细介绍这些RTOS的特点。以下内容来自官方网站或官方手册Feature的谷歌翻译,由我进行了一些调整,没有任何主观因素。1.FreeRTOS FreeRTOS是一个为小型嵌入式系统设计的可扩展实时内核。亮点包括:小包装。免费RTOS调度器免费嵌入式软件源代码。免版税。抢占式、协作式和混合式配置选项以及时间切片是可选的。SafeRTOS衍生产品提供高...