【codeforces 528D】 Fuzzy Search
http://codeforces.com/problemset/problem/528/D (题目链接)
题意
给定母串和模式串,字符集大小为${4}$,给定${k}$,模式串在某个位置匹配当且仅当任意位置模式串的这个字符所对应的母串的位置的左右${k}$个字符之内有一个与它相同的,求匹配次数。
Solution
毛爷爷论文题。我们将${4}$种不同的字符分开计算贡献。每一次计算,先预处理出母串种的每个位置能否匹配,对于每个能够匹配的位置,我们将它赋为${1}$,不能匹配则赋为${0}$,将其存放在数组${A}$中。对于模式串,如果它这一位等于当前计算的这个字符,就将这一位赋为${1}$,否则赋为${0}$,将其存放在数组${B}$中。这样的话,我们将模式串倒过来,那么最后对于这个字符来说,模式串位于某一位置与主串的匹配字符数量就是${A}$和${B}$的卷积。那么最后将所有${4}$种字符的贡献算完,如果某一位置的卷积等于模式串长度,那么久意味着模式串处于这一位置的时候可以与主串匹配。于是我们就可以用${FFT}$来解决这个问题了。
细节
${FFT}$最后统计的时候要转成${LL}$而不是${int}$。
代码
// codeforces 528D
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<complex>
#include<cstring>
#include<cstdio>
#include<cmath>
#define LL long long
#define inf 2147483640
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std; typedef complex<double> E;
const int maxn=800010;
int n,m,N,M,K,L;
int ans[maxn],rev[maxn],id[maxn],cnt[5],vis[maxn][5];
char a[maxn],b[maxn];
E A[maxn],B[maxn]; void FFT(E *a,int f) {
for (int i=0;i<N;i++) if (rev[i]>i) swap(a[i],a[rev[i]]);
for (int i=1;i<N;i<<=1) {
E wn(cos(Pi/i),sin(Pi/i));
for (int p=i<<1,j=0;j<N;j+=p) {
E w(1,0);
for (int k=0;k<i;k++,w*=wn) {
E x=a[k+j],y=w*a[k+j+i];
a[k+j]=x+y;a[k+j+i]=x-y;
}
}
}
if (f==-1) reverse(a+1,a+N);
}
int main() {
scanf("%d%d%d",&n,&m,&K);
scanf("%s%s",a+1,b+1);
id['A']=1;id['T']=2;id['G']=3;id['C']=4;
for (int l=0,r=0,i=1;i<=n;i++) { //get了一个比较美观的写法
while (l<n && l<i-K) cnt[id[(int)a[l++]]]--;
while (r<n && r<i+K) cnt[id[(int)a[++r]]]++;
for (int j=1;j<=4;j++) if (cnt[j]) vis[i][j]=1;
}
M=n+m;
for (N=1;N<=M;N<<=1) L++;
for (int i=0;i<N;i++) rev[i]=(rev[i>>1]>>1) | ((i&1)<<(L-1));
for (int k=1;k<=4;k++) {
memset(A,0,sizeof(A));
memset(B,0,sizeof(B));
for (int i=1;i<=n;i++) {
if (vis[i][k]) A[i-1]=1;
else A[i-1]=0;
}
for (int i=1;i<=m;i++) {
if (id[(int)b[i]]==k) B[m-i]=1;
else B[m-i]=0;
}
FFT(A,1);FFT(B,1);
for (int i=0;i<N;i++) A[i]*=B[i];
FFT(A,-1);
for (int i=0;i<N;i++) ans[i]+=(LL)(A[i].real()+0.5)/N; //此处LL
}
int res=0;
for (int i=0;i<N;i++) if (ans[i]==m) res++;
printf("%d",res);
return 0;
}
【codeforces 528D】 Fuzzy Search的更多相关文章
- 【CF528D】Fuzzy Search(FFT)
		
[CF528D]Fuzzy Search(FFT) 题面 给定两个只含有\(A,T,G,C\)的\(DNA\)序列 定义一个字符\(c\)可以被匹配为:它对齐的字符,在距离\(K\)以内,存在一个字符 ...
 - 【codeforces 415D】Mashmokh and ACM(普通dp)
		
[codeforces 415D]Mashmokh and ACM 题意:美丽数列定义:对于数列中的每一个i都满足:arr[i+1]%arr[i]==0 输入n,k(1<=n,k<=200 ...
 - 【Codeforces528D】Fuzzy Search      FFT
		
D. Fuzzy Search time limit per test:3 seconds memory limit per test:256 megabytes input:standard inp ...
 - 【codeforces 707E】Garlands
		
[题目链接]:http://codeforces.com/contest/707/problem/E [题意] 给你一个n*m的方阵; 里面有k个联通块; 这k个联通块,每个连通块里面都是灯; 给你q ...
 - 【codeforces 707C】Pythagorean Triples
		
[题目链接]:http://codeforces.com/contest/707/problem/C [题意] 给你一个数字n; 问你这个数字是不是某个三角形的一条边; 如果是让你输出另外两条边的大小 ...
 - 【codeforces 709D】Recover the String
		
[题目链接]:http://codeforces.com/problemset/problem/709/D [题意] 给你一个序列; 给出01子列和10子列和00子列以及11子列的个数; 然后让你输出 ...
 - 【codeforces 709B】Checkpoints
		
[题目链接]:http://codeforces.com/contest/709/problem/B [题意] 让你从起点开始走过n-1个点(至少n-1个) 问你最少走多远; [题解] 肯定不多走啊; ...
 - 【codeforces 709C】Letters Cyclic Shift
		
[题目链接]:http://codeforces.com/contest/709/problem/C [题意] 让你改变一个字符串的子集(连续的一段); ->这一段的每个字符的字母都变成之前的一 ...
 - 【Codeforces 429D】 Tricky Function
		
[题目链接] http://codeforces.com/problemset/problem/429/D [算法] 令Si = A1 + A2 + ... + Ai(A的前缀和) 则g(i,j) = ...
 
随机推荐
- 4星|《财经》2018年第10期:远程视界自我定位为“专科远程医疗联合体O2O平台”,主要盈利模式就是做融资租赁
			
<财经>2018年第10期 总第527期 旬刊 本期主要内容:做远程医疗资金链断裂:人工智能时代有可能让刘易斯观点论失败:小米的盈利模式刨析:陆奇在百度的改革.其中1.4都成了朋友圈热文. ...
 - gitlab+jenkins持续集成(二)
			
1.jenkins服务器上的配置 -bin.tar.gz -C /opt/ yum install -y git /conf/settings.xml #只需更改maven的地址 <?xml v ...
 - Hyperledger Fabric CA User’s Guide——配置设置(四)
			
配置设置 Fabric CA提供了三种方案去配置Fabric CA服务端和客户端,优先顺序是: CLI flags(标识) 环境变量 配置文件 在本文档的其余部分中,我们将对配置文件进行更改.但是,可 ...
 - nodejs 中jead模板改为ejs
			
var app = express(); // view engine setup app.set('views', path.join(__dirname, 'views')); app.set(' ...
 - virtual box下安装ubuntu经验
			
1. 哪怕下载的是ubuntu64位版本,也在vitualbox下选择ubuntu而不要选择ubuntu(64bit) 2. 安装VBoxGuestAdditional.iso:下载和vbox版本相匹 ...
 - 第十二次ScrumMeeting博客
			
第十二次ScrumMeeting博客 本次会议于11月30日(四)22时整在3公寓725房间召开,持续35分钟. 与会人员:刘畅.辛德泰.张安澜.赵奕.方科栋. 1. 每个人的工作(有Issue的内容 ...
 - 调试存储过程:ORA-0131 Insufficient privileges
			
http://www.cnblogs.com/empty01/p/5568250.html
 - vim 多个文件切换  :b 命令
			
MiniBufExplorer插件的使用 博客分类: vim vimMiniBufExplorer 快速浏览和操作Buffer -- 插件: MiniBufExplorer 下载地址 [http:// ...
 - Linux读书笔记第一、二章
			
第一章 Linux内核简介 1.1Unix历史 Unix特点:1.很简洁 2.所有东西都被当成文件对待 3.Unix内核和相关的系统工具软件都是用C语言编写而成 4.进程创建非常迅速 1.2追寻 ...
 - 20135313-exp2
			
北京电子科技学院(BESTI) 实 验 报 告 课程:Java程序设计 班级:1353 姓名:吴子怡(20135313) 成绩: 指导教师:娄嘉鹏 实验日期 ...