原文链接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 

Output 

Input  

Output
        

2. fjzzq2002 的博客中给出的数据。

相应博客链接见“参考文献”。

参考文献

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

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

Berlekamp_Massey 算法 (BM算法) 学习笔记的更多相关文章

  1. 牛客网《BAT面试算法精品课》学习笔记

    目录 牛客网<BAT面试算法精品课>学习笔记 牛客网<BAT面试算法精品课>笔记一:排序 牛客网<BAT面试算法精品课>笔记二:字符串 牛客网<BAT面试算法 ...

  2. 数论算法 剩余系相关 学习笔记 (基础回顾,(ex)CRT,(ex)lucas,(ex)BSGS,原根与指标入门,高次剩余,Miller_Rabin+Pollard_Rho)

    注:转载本文须标明出处. 原文链接https://www.cnblogs.com/zhouzhendong/p/Number-theory.html 数论算法 剩余系相关 学习笔记 (基础回顾,(ex ...

  3. BZOJ 2038: [2009国家集训队]小Z的袜子(hose)【莫队算法裸题&&学习笔记】

    2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 9894  Solved: 4561[Subm ...

  4. 【数据结构与算法Python版学习笔记】引言

    学习来源 北京大学-数据结构与算法Python版 目标 了解计算机科学.程序设计和问题解决的基本概念 计算机科学是对问题本身.问题的解决.以及问题求解过程中得出的解决方案的研究.面对一 个特定问题,计 ...

  5. 【数据结构与算法Python版学习笔记】图——强连通分支

    互联网 我们关注一下互联网相关的非常巨大图: 由主机通过网线(或无线)连接而形成的图: 以及由网页通过超链接连接而形成的图. 网页形成的图 以网页(URI作为id)为顶点,网页内包含的超链接作为边,可 ...

  6. 数据结构与算法C++描述学习笔记1、辗转相除——欧几里得算法

    前面学了一个星期的C++,以前阅读C++代码有些困难,现在好一些了.做了一些NOI的题目,这也是一个长期的目标中的一环.做到动态规划的相关题目时发现很多问题思考不通透,所以开始系统学习.学习的第一本是 ...

  7. 平滑处理Smooth之图像预处理算法-OpenCV应用学习笔记三

    大清早的我们就来做一个简单有趣的图像处理算法实现,作为对图像处理算法学习的开端吧.之所以有趣就在于笔者把算法处理的各个方式的处理效果拿出来做了对比,给你看到原图和各种处理后的图像你是否能够知道那幅图对 ...

  8. CNN目标检测系列算法发展脉络——学习笔记(一):AlexNet

    在咨询了老师的建议后,最近开始着手深入的学习一下目标检测算法,结合这两天所查到的资料和个人的理解,准备大致将CNN目标检测的发展脉络理一理(暂时只讲CNN系列部分,YOLO和SSD,后面会抽空整理). ...

  9. Canny边缘检测及图像缩放之图像处理算法-OpenCV应用学习笔记四

    在边缘检测算法中Canny颇为经典,我们就来做一下测试,并且顺便实现图像的尺寸放缩. 实现功能: 直接执行程序得到结果如下:将载入图像显示在窗口in内,同时进行图像两次缩小一半操作将结果显示到i1,i ...

随机推荐

  1. 爬虫与request模块

    一.爬虫简介 1.介绍 网络爬虫(又被称为网页蜘蛛,网络机器人,在FOAF社区中间,更经常的称为网页追逐者),是一种按照一定的规则,自动地抓取万维网信息的程序或者脚本.另外一些不常使用的名字还有蚂蚁. ...

  2. python 实现进制转换(二进制转十进制)

    摘自https://baike.baidu.com/item/%E5%8D%81%E8%BF%9B%E5%88%B6%E8%BD%AC%E4%BA%8C%E8%BF%9B%E5%88%B6 pytho ...

  3. Qt: 非阻塞时间延迟;

    1.使用时间耗损循环: #include <QTime> ... QTime delayTime = QTime::currentTime().addMSecs(1000); while( ...

  4. ubuntu:基本操作;

    1.系统时间与网络时间同步: sudo dpkg-reconfigure tzdata 注: 该博文为扩展型:

  5. SpringBoot系列: Spring支持的异常处理方式

    ===================================视图函数返回 status code 的方式===================================Spring 有 ...

  6. 小程序图片转Base64

    在小程序中,有些业务要用到 图片的 base64 wx.chooseImage({ success: res => { wx.getFileSystemManager().readFile({ ...

  7. VIM --使用进阶 -- 插件篇 -- YouCompleteMe -- nerdtree

    系统:ubuntu: 资源:https://github.com/ 其他:想了解都要哪些好用的插件,推荐大家读 http://blog.csdn.net/mergerly/article/detail ...

  8. nnet3配置中的上下文和chunk(块)大小

    Nnet3配置中的上下文和块大小 简介 本页讨论了nnet3配置中关于解码和训练的块大小以及左右上下文的某些术语.这将有助于理解一些脚本.目前,从脚本角度来看,没有任何关于nnet3的"概述 ...

  9. 程序设计-理解java继承-遁地龙卷风

    (0)写在前面 编程和现实世界是息息相关的,你是如何理解现实世界中的继承呢? 继承在编程中应理解为:对父类资源(本文只讨论方法)的使用,父类方法只提供类基本的功能,父类方法之间不存在调用关系. (1) ...

  10. Ansible------常用功能

    local_action Ansible 默认只会对控制机器执行操作,但如果在这个过程中需要在 Ansible 本机执行操作就需要使用到local_action become:True Ansible ...