CodeForces 528D Fuzzy Search(FFT)

题目链接

之前说了怎么用FFT匹配字符串,这题也差不多
显然只有ACGT这么几个是在暗示我们把他们分开算
长串t,短串s,长串长lenb,短串长lena
那么我们的任务就变成了计算在从长串t的p位置开始能不能把短串里的所有这类字符匹配上去
我们定义一个函数f[i]
f[i]=1表示长串t中i这个位置左右k内都没有该字符
f[i]=0表示有
函数g[i]表示i这个位置是不是该字符
显然匹配失败当且仅当f[p+i]=1并且g[i]=1,所以匹配成功的条件是
∑ i = 0 l e n a f [ p + i ] g [ i ] = 0 \sum_{i=0}^{lena}f[p+i]g[i]=0 i=0lenaf[p+i]g[i]=0
那么显然把g反转一下,就又变成了卷积的形式
分别求然后FFT就可以了。

代码如下:

#include<bits/stdc++.h>
#define mod 950009857
#define gg 7
using namespace std;

string m="ATGC";
long long f[600040],g[600040];
int r[600060],ans[600040],nxt[600040],pre[600040];
int lim,k;
char s[200020],t[200020];

long long kasumi(long long a,long long b)
{
	long long ans=1;
	while(b)
	{
		if(b&1) ans=ans*a%mod;
		a=a*a%mod;
		b>>=1;
	}
	return ans;
}

void NTT(long long *a,int kd)
{
	for(int i=0;i<lim;i++) 
	{
		if(i<r[i]) swap(a[i],a[r[i]]);
	}
	for(int mid=1;mid<lim;mid<<=1)
	{
		long long wn=kasumi(gg,(mod-1)/(mid<<1));
		if(kd) wn=kasumi(wn,mod-2);
		for(int i=0;i<lim;i+=(mid<<1))
		{
			long long w=1;
			for(int j=0;j<mid;j++,w=wn*w%mod)
			{
				long long x=a[i+j];
				long long y=a[i+j+mid]*w%mod;
				a[i+j]=(x+y)%mod;
				a[i+j+mid]=(x-y+mod)%mod;
			}
		}
	}
	if(kd)
	{
		long long inv1=kasumi(lim,mod-2);
		for(int i=0;i<lim;i++)
		{
			a[i]=a[i]*inv1%mod;
		}
	}
}

int cnt=0,lena,lenb;

int main()
{
	scanf("%d%d",&lena,&lenb);
	scanf("%d",&k);
	scanf("%s",s);
	scanf("%s",t);
	for(int i=0;i<=lena-lenb;i++) ans[i]=1;
	for(lim=1;lim<=lena+lenb;lim<<=1,cnt++);
	for(int i=0;i<lim;i++)
	{
		r[i]=(r[i>>1]>>1)|((i&1)<<(cnt-1));
	}
	for(int ah=0;ah<4;ah++)
	{
		memset(f,0,sizeof(f));
		memset(g,0,sizeof(g));
		char nowc=m[ah];
		for(int i=0;i<lenb;i++)
		{
			g[i]=(t[i]==nowc);
		}
		reverse(g,g+lim);
		int pr=-200000;
		for(int i=0;i<lena;i++)
		{
			if(s[i]==nowc) pr=i;
			pre[i]=pr;
		}
		int nx=1000000;
		for(int i=lena-1;i>=0;i--)
		{
			if(s[i]==nowc) nx=i;
			nxt[i]=nx;
		}
		for(int i=0;i<lena;i++)
		{
			if(i-pre[i]>k&&nxt[i]-i>k) f[i+1]=1;
			else f[i+1]=0;
		}
		NTT(f,0);NTT(g,0);
		for(int i=0;i<lim;i++)
		{
			f[i]=f[i]*g[i]%mod;
		}
		NTT(f,1);
		for(int i=0;i<=lena-lenb;i++)
		{
			ans[i]=ans[i]&(!f[i]);
		}
	}
	long long res=0;
	for(int i=0;i<=lena-lenb;i++) if(ans[i]) res++;
	printf("%lld\n",res);
}
基于bert实现关系三元组抽取python源码+数据集+项目说明.zip基于bert实现关系三元组抽取python源码+数据集+项目说明.zip基于bert实现关系三元组抽取python源码+数据集+项目说明.zip基于bert实现关系三元组抽取python源码+数据集+项目说明.zip基于bert实现关系三元组抽取python源码+数据集+项目说明.zip 个人大四的毕业设计、课程设计、作业、经导师指导并认可通过的高分设计项目,评审平均分达96.5分。主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者,也可作为课程设计、期末大作业。 [资源说明] 不懂运行,下载完可以私聊问,可远程教学 该资源内项目源码是个人的毕设或者课设、作业,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96.5分,放心下载使用! 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),供学习参考。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值