参考于:

https://www.luogu.org/problemnew/solution/P4723 shadowice1984 (太难)

https://www.cnblogs.com/zhgyki/p/9671855.html (玄学)

居然还有解释:

https://www.cnblogs.com/zhouzhendong/p/Berlekamp-Massey.html

线性递推BM算法,把前面的8~10项push进去?

下面是板子,求第n项调用参数是n-1。

#include<bits/stdc++.h>
using namespace std;
#define rep(i,a,n) for (int i=a;i<n;i++)
#define per(i,a,n) for (int i=n-1;i>=a;i--)
#define pb push_back
#define mp make_pair
#define all(x) (x).begin(),(x).end()
#define fi first
#define se second
#define SZ(x) ((int)(x).size())
typedef vector<int> VI;
typedef long long ll;
typedef 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;}
// head ll n;
namespace linear_seq {
const int N=10010;
ll res[N],base[N],_c[N],_md[N]; vector<int> Md;
void mul(ll *a,ll *b,int k) {
rep(i,0,k+k) _c[i]=0;
rep(i,0,k) if (a[i]) rep(j,0,k) _c[i+j]=(_c[i+j]+a[i]*b[j])%mod;
for (int i=k+k-1;i>=k;i--) if (_c[i])
rep(j,0,SZ(Md)) _c[i-k+Md[j]]=(_c[i-k+Md[j]]-_c[i]*_md[Md[j]])%mod;
rep(i,0,k) a[i]=_c[i];
}
int solve(ll n,VI a,VI b) { // a 系数 b 初值 b[n+1]=a[0]*b[n]+...
ll ans=0,pnt=0;
int k=SZ(a);
assert(SZ(a)==SZ(b));
rep(i,0,k) _md[k-1-i]=-a[i];_md[k]=1;
Md.clear();
rep(i,0,k) if (_md[i]!=0) Md.push_back(i);
rep(i,0,k) 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;
rep(j,0,SZ(Md)) res[Md[j]]=(res[Md[j]]-res[k]*_md[Md[j]])%mod;
}
}
rep(i,0,k) 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;
rep(n,0,SZ(s)) {
ll d=0;
rep(i,0,L+1) 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 (SZ(C)<SZ(B)+m) C.pb(0);
rep(i,0,SZ(B)) 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 (SZ(C)<SZ(B)+m) C.pb(0);
rep(i,0,SZ(B)) 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());
rep(i,0,SZ(c)) c[i]=(mod-c[i])%mod;
return solve(n,c,VI(a.begin(),a.begin()+SZ(c)));
}
}; int main() {
/*push_back 进去前 8~10 项左右、最后调用 gao 得第 n 项*/
vector<int>v;
v.push_back(3);
v.push_back(9);
v.push_back(20);
v.push_back(46);
v.push_back(106);
v.push_back(244);
v.push_back(560);
v.push_back(1286);
v.push_back(2956);
v.push_back(6794);
int nCase;
scanf("%d", &nCase);
while(nCase--){
scanf("%lld", &n);
printf("%lld\n",1LL * linear_seq::gao(v,n-1) % mod);
}
}

仿照上面的思路瞎搞通过。

#include<bits/stdc++.h>
using namespace std;
#define rep(i,a,n) for (int i=a;i<n;i++)
typedef vector<int> vi;
typedef long long ll;
const ll mod=1000000007; ll qpow(ll x,ll n) {
ll res=1;
for(; n; x=x*x%mod,n>>=1)
if(n&1)
res=res*x%mod;
return res;
} namespace Linear_Recurrence { //空间无所谓,开大一点
const int N=10010;
ll res[N],base[N],c[N],md[N]; int umd[N],sizumd;
void mul(ll *a,ll *b,int k) {
rep(i,0,k+k) c[i]=0;
rep(i,0,k) if (a[i])
rep(j,0,k) c[i+j]=(c[i+j]+a[i]*b[j])%mod; for (int i=k+k-1; i>=k; i--)
if (c[i])
rep(j,0,sizumd) c[i-k+umd[j]]=(c[i-k+umd[j]]-c[i]*md[umd[j]])%mod; rep(i,0,k) a[i]=c[i];
} int solve(ll n,vi a,vi b) {
//a是系数向量,b是初值向量
//递推公式形如b[n+1]=a[0]*b[n]+a[1]*b[n-1]+...
ll ans=0,pnt=0;
int k=a.size();
rep(i,0,k) md[k-1-i]=-a[i];
md[k]=1;
sizumd=0;
rep(i,0,k) {
if (md[i]!=0)
umd[sizumd++]=i;
}
rep(i,0,k) res[i]=0,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;
rep(j,0,sizumd) res[umd[j]]=(res[umd[j]]-res[k]*md[umd[j]])%mod;
}
}
rep(i,0,k) 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;
int sizs=s.size();
rep(n,0,sizs) {
ll d=0;
rep(i,0,L+1) d=(d+(ll)C[i]*s[n-i])%mod;
if (d==0)
++m;
else if (2*L<=n) {
vi T=C;
int sizB=B.size();
while (C.size()<sizB+m)
C.push_back(0);
ll t=mod-d*qpow(b,mod-2)%mod;
rep(i,0,sizB) C[i+m]=(C[i+m]+t*B[i])%mod;
L=n+1-L,B=T,b=d,m=1;
} else {
int sizB=B.size();
while (C.size()<sizB+m)
C.push_back(0);
ll t=mod-d*qpow(b,mod-2)%mod;
rep(i,0,sizB) C[i+m]=(C[i+m]+t*B[i])%mod;
++m;
}
}
return C;
}
int gao(vi A,ll n) {
vi C=BM(A);
C.erase(C.begin());
int sizC=C.size();
rep(i,0,sizC) C[i]=(mod-C[i])%mod;
return solve(n,C,vi(A.begin(),A.begin()+sizC));
}
}; ll dp[3005]; ll solve(int k,ll n) {
ll invk=qpow(k,mod-2);
dp[0]=1;
//至少取2倍以上
int c=k*2;
for(int i=1; i<=c+1; i++) {
dp[i]=0;
for(int j=max(0,i-k); j<=i-1; j++) {
dp[i]+=dp[j];
}
dp[i]=(dp[i]%mod)*invk%mod;
}
vi v;
for(int i=0; i<=c+1; i++)
v.push_back((int)dp[i]);
//从0开始,到n,这里其实是第n+1项
return Linear_Recurrence::gao(v,n);
} int main() {
#ifdef Yinku
freopen("Yinku.in","r",stdin);
#endif // Yinku
int t,k;
ll n;
scanf("%d", &t);
while(t--) {
scanf("%d%lld", &k,&n);
if(n==-1)
printf("%lld\n",2ll*qpow(k+1,mod-2)%mod);
else
printf("%lld\n",solve(k,n));
}
}

好像前2K项比较保险,前K项就WA了,前1.5K项也WA了,临界值是多少呢?反正大概取2~3倍递推可能就够了吧。

2019牛客暑期多校训练营(第二场) - B - Eddy Walker 2 - BM算法的更多相关文章

  1. 2019牛客暑期多校训练营(第二场) H-Second Large Rectangle(单调栈)

    题意:给出由01组成的矩阵,求求全是1的次大子矩阵. 思路: 单调栈 全是1的最大子矩阵的变形,不能直接把所有的面积存起来然后排序取第二大的,因为次大子矩阵可能在最大子矩阵里面,比如: 1 0 0 1 ...

  2. 2020牛客暑期多校训练营 第二场 K Keyboard Free 积分 期望 数学

    LINK:Keyboard Free 我要是会正经的做法 就有鬼了. 我的数学水平没那么高. 三个同心圆 三个动点 求围成三角形面积的期望. 不会告辞. 其实可以\(n^2\)枚举角度然后算出面积 近 ...

  3. 2020牛客暑期多校训练营 第二场 J Just Shuffle 置换 群论

    LINK:Just Shuffle 比较怂群论 因为没怎么学过 置换也是刚理解. 这道题是 已知一个置换\(A\)求一个置换P 两个置换的关键为\(P^k=A\) 且k是一个大质数. 做法是李指导教我 ...

  4. 2020牛客暑期多校训练营 第二场 I Interval 最大流 最小割 平面图对偶图转最短路

    LINK:Interval 赛时连题目都没看. 观察n的范围不大不小 而且建图明显 考虑跑最大流最小割. 图有点稠密dinic不太行. 一个常见的trick就是对偶图转最短路. 建图有点复杂 不过建完 ...

  5. 2020牛客暑期多校训练营 第二场 C Cover the Tree 构造 贪心

    LINK:Cover the Tree 最受挫的是这道题,以为很简单 当时什么都想不清楚. 先胡了一个树的直径乱搞的贪心 一直过不去.后来意识到这类似于最经典长链剖分优化贪心的做法 然后那个是求最大值 ...

  6. 2020牛客暑期多校训练营 第二场 B Boundary 计算几何 圆 已知三点求圆心

    LINK:Boundary 计算几何确实是弱项 因为好多东西都不太会求 没有到很精通的地步. 做法很多,先说官方题解 其实就是枚举一个点 P 然后可以发现 再枚举一个点 然后再判断有多少个点在圆上显然 ...

  7. 2020牛客暑期多校训练营 第二场 A All with Pairs 字符串hash KMP

    LINK:All with Pairs 那天下午打这个东西的时候状态极差 推这个东西都推了1个多小时 (比赛是中午考试的我很困 没睡觉直接开肝果然不爽 一开始看错匹配的位置了 以为是\(1-l\)和\ ...

  8. 2019牛客暑期多校训练营(第九场) D Knapsack Cryptosystem

    题目 题意: 给你n(最大36)个数,让你从这n个数里面找出来一些数,使这些数的和等于s(题目输入),用到的数输出1,没有用到的数输出0 例如:3  4 2 3 4 输出:0 0 1 题解: 认真想一 ...

  9. 2019牛客暑期多校训练营(第五场)G - subsequeue 1 (一题我真的不会的题)

    layout: post title: 2019牛客暑期多校训练营(第五场)G - subsequeue 1 (一题我真的不会的题) author: "luowentaoaa" c ...

  10. 2019牛客暑期多校训练营(第二场)F.Partition problem

    链接:https://ac.nowcoder.com/acm/contest/882/F来源:牛客网 Given 2N people, you need to assign each of them ...

随机推荐

  1. 9.Spring Cloud Config统一管理微服务配置

    Spring Cloud Config统一管理微服务配置 9.1. 为什么要统一管理微服务配置 9.2. Spring Cloud Config简介 Spring Cloud Config为分布式系统 ...

  2. 20.包含min函数的栈 Java

    题目描述 定义栈的数据结构,请在该类型中实现一个能够得到栈中所含最小元素的min函数(时间复杂度应为O(1)). 思路 借助辅助栈实现: 压栈时:若辅助栈为空,则将节点压入辅助栈.否则,当当前节点小于 ...

  3. 【零基础】快速入门爬虫框架HtmlUnit

    迅速的HtmlUnit htmlunit是一款开源的web页面分析工具,理论上来说htmlunit应用于网页的自动化测试,但是相对来说更多人使用它来进行小型爬虫的快速开发.使用htmlunit进行爬虫 ...

  4. Netfilter 之 连接跟踪初始化

    基础参数初始化 nf_conntrack_init_start函数完成连接跟踪基础参数的初始化,包括了hash,slab,扩展项,GC任务等: int nf_conntrack_init_start( ...

  5. LeetCode 200. 岛屿的个数(Number of Islands)

    题目描述 给定一个由 '1'(陆地)和 '0'(水)组成的的二维网格,计算岛屿的数量.一个岛被水包围,并且它是通过水平方向或垂直方向上相邻的陆地连接而成的.你可以假设网格的四个边均被水包围. 示例 1 ...

  6. java调用exe,及调用bat不成功的解决办法

    开门见山的说,文件目录如下 想调用123.exe,但是尝试了几次调用不到,然后写了个bat.初始内容如下 @echo off D: cd test "123.exe" 双击可以运行 ...

  7. 性能测试 | 记一次生产数据库sql由451s优化为0.4s的过程

    概述 最近开发说某个接口跑的很慢,排查了下发现其中一条sql,数据量不大,但居然要跑451s,下面简单记录一下优化的过程. 问题sql SELECT l.location_gid ENUMVALUE, ...

  8. go module 使用举例

    go语言中,从1.11开始,引入module,进行版本管理. 通过使用module,工程目录的位置不用必须放在GOPATH下. 本文介绍 module的使用. 下文中用的Go版本是1.13. 1. g ...

  9. smb服务问题解析,区别红帽6和CentOS7

    最近在学习smb服务,哇!简直问题多的不要不要的.因为以前是学习的红帽6的系统,现在用的是CentOs7,所以还是改不了以前的一些配置方式,造成了很多问题.快成地中海了! 我们准备环境: 系统: 服务 ...

  10. easyUI之练习

    <%@ page language="java" pageEncoding="UTF-8"%> <!DOCTYPE HTML PUBLIC & ...