「LibreOJ β Round #7」匹配字符串
一、题目
题目质量确实高,我要开始进军 了。
二、解法
很容易写出本题的 方程,设 表示考虑前 个位置合法,第 个位置是 的方案数:
这很显然是个常系数齐次线性递推,我们求出 的多项式 ,那么答案是:
因为 的系数很规则,具体实现中可以不用多项式取模,卷积完之后直接倒序做差分即可。那么时间复杂度 ,这种做法可以在 较小的时候派上用场。
我们还需要一种算法能解决 较大的情况,设 ,那么 ,他的组合意义是从 走到 ,有 到 的权值为 的边,有 到 的权值为 的边。定义一种路径的权值为经过的所有边的权值乘积,那么答案是所有路径的权值和,我们枚举 边的数量:
其中 就是选出 条 边的方案数,相邻两条边的距离必须超过 ,上面柿子从容斥的角度也能解释,也就是我们钦定 个不合法段的起点,用 暴力计算上式,再结合算法 即可。
三、总结
思考 的组合意义很重要,这个技巧我以前起了个名字叫考虑转移路径。
当获得了一种很优秀的做法之后,可以考虑另一种做法来数据分治解决问题。
#include <cstdio>
#include <iostream>
using namespace std;
const int M = 100005;
const int MOD = 65537;
#define int long long
int read()
{
int x=0,f=1;char c;
while((c=getchar())<'0' || c>'9') {if(c=='-') f=-1;}
while(c>='0' && c<='9') {x=(x<<3)+(x<<1)+(c^48);c=getchar();}
return x*f;
}
int n,m,ans;
//subtask1
int a[M],r[M],A[M],B[M],rev[M];
int qkpow(int a,int b)
{
int r=1;
while(b>0)
{
if(b&1) r=r*a%MOD;
a=a*a%MOD;
b>>=1;
}
return r;
}
void NTT(int *a,int len,int op)
{
for(int i=0;i<len;i++)
{
rev[i]=(rev[i>>1]>>1)|((i&1)*(len/2));
if(i<rev[i]) swap(a[i],a[rev[i]]);
}
for(int s=2;s<=len;s<<=1)
{
int w=(op==1)?qkpow(3,(MOD-1)/s):qkpow(3,(MOD-1)-(MOD-1)/s);
for(int i=0,t=s/2;i<len;i+=s)
for(int j=0,x=1;j<t;j++,x=x*w%MOD)
{
int fo=a[i+j],fe=a[i+j+t];
a[i+j]=(fo+x*fe)%MOD;
a[i+j+t]=((fo-x*fe)%MOD+MOD)%MOD;
}
}
if(op==1) return ;
int inv=qkpow(len,MOD-2);
for(int i=0;i<len;i++)
a[i]=a[i]*inv%MOD;
}
void mul(int *a,int *b,int *c)
{
int len=1;
while(len<2*m) len<<=1;
for(int i=0;i<len;i++) A[i]=0;
for(int i=0;i<len;i++) B[i]=0;
for(int i=0;i<m;i++) A[i]=a[i];
for(int i=0;i<m;i++) B[i]=b[i];
NTT(A,len,1);NTT(B,len,1);
for(int i=0;i<len;i++) A[i]=A[i]*B[i]%MOD;
NTT(A,len,-1);
for(int i=0;i<len;i++) B[i]=0;
for(int i=2*m-2;i>=m;i--)
{
B[i]=(B[i]+B[i+1])%MOD;
A[i]=(A[i]+B[i])%MOD;
B[i-1]=(B[i-1]+A[i])%MOD;
if(i>m) B[i-1-m]=(B[i-1-m]-A[i]+MOD)%MOD;
}
for(int i=m-1;i>=0;i--)
{
B[i]=(B[i]+B[i+1])%MOD;
c[i]=(A[i]+B[i])%MOD;
}
}
void work1()
{
if(m==1)
{
puts("1");
return ;
}
a[1]=1;r[0]=1;
while(n>0)
{
if(n&1) mul(r,a,r);
mul(a,a,a);
n>>=1;
}//x^n
for(int i=0,pw=1;i<m;i++,pw=pw*2%MOD)
ans=(ans+pw*r[i])%MOD;
printf("%lld\n",ans);
}
//subtask2
int fac[M],inv[M];
void init(int n)
{
fac[0]=inv[0]=inv[1]=1;
for(int i=2;i<=n;i++) inv[i]=inv[MOD%i]*(MOD-MOD/i)%MOD;
for(int i=1;i<=n;i++) inv[i]=inv[i]*inv[i-1]%MOD;
for(int i=1;i<=n;i++) fac[i]=fac[i-1]*i%MOD;
}
int C(int n,int m)
{
if(n<m || m<0) return 0;
return fac[n]*inv[m]%MOD*inv[n-m]%MOD;
}
int lucas(int n,int m)
{
if(n<=m) return n==m;
return lucas(n/MOD,m/MOD)*C(n%MOD,m%MOD)%MOD;
}
int work2(int n)
{
init(MOD-1);
int ans=0,fl=1,pw=qkpow(2,n),iv=qkpow(inv[2],m+1);
for(int i=0;i<=n/(m+1);i++)
{
ans=(ans+fl*pw%MOD*lucas(n-i*m,i))%MOD;
pw=pw*iv%MOD;fl=MOD-fl;
}
return ans;
}
signed main()
{
n=read();m=read();
if(m<1<<15) work1();
else
{
init(MOD-1);
printf("%lld\n",(work2(n+1)-work2(n)+MOD)%MOD);
}
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 地球OL攻略 —— 某应届生求职总结
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 提示词工程——AI应用必不可少的技术
· .NET周刊【3月第1期 2025-03-02】