CF_528D
一句话题意
给你两个串s、t,长度为n、m,字符集为"ATGC",当且仅
当[i - k; i + k]中存在一个j,使得s[j ] = t[x]时,s[i ]可以
和t[x]匹配,问t总共能与s的几个子串匹配
首先,字符集只有4,那么,令s_A[i]=0/1 表示在s中i位置能否和A匹配,同理t_A[i];
一类关于通配符匹配的字符串问题都是可以用FFT来解决的
以A字符为例,令
\]
则当c[i]==D时 以S[i]为首的一个子串内的A的位置与T的A的位置相同
令
\]
代入原式,则
\]
成为了卷积形式
P.S. 也可以用bitset
#include<bits/stdc++.h>
using namespace std;
namespace Tzh{
typedef double dd;
const char ch[]="0ATCG";
const int maxn=700010;
const dd pi=acos(-1);
int ans,vis[maxn],x,y,k,rev[maxn],n=1,len,sum;
string S,T;
struct complex{
dd x,y;
complex operator +(const complex &b) const{
return (complex){x+b.x,y+b.y};
}
complex operator -(const complex &b) const{
return (complex){x-b.x,y-b.y};
}
complex operator *(const complex &b) const{
return (complex){x*b.x-y*b.y,x*b.y+y*b.x};
}
void init(){
x=0,y=0;
}
}s[maxn],t[maxn];
void init(){
for(int i=0;i<n;i++)
rev[i]=rev[i>>1]>>1|((i&1)<<(len-1));
}
complex omg(int x,int flag){
return (complex){cos((dd)x*2*pi/n),(dd)sin((dd)x*2*pi/n)*flag};
}
void FFT(complex *a,int type){
for(int i=0;i<n;i++) if(rev[i]>i) swap(a[i],a[rev[i]]);
for(int l=2,m=1;l<=n;m=l,l<<=1)
for(int i=0;i<n;i+=l)
for(int j=0;j<m;j++){
complex tt=omg(n/l*j,type)*a[i+j+m];
a[i+j+m]=a[i+j]-tt,a[i+j]=a[i+j]+tt;
}
if(type==-1)
for(int i=0;i<n;i++) a[i].x/=n;
}
void work(){
ios::sync_with_stdio(false);
cin>>x>>y>>k;
cin>>S>>T; for(int i=0;i<=x;i++) vis[i]=1;
while(n<=x+y) n<<=1,len++; init();
for(int j=1;j<=4;j++){ int num=0;
for(int i=0;i<S.size();i++){
if(S[i]==ch[j]) num++;
if(i>k&&S[i-k-1]==ch[j]) num--;
s[i].x=(dd)(num>0);
} num=0,sum=0;
for(int i=S.size()-1;i>=0;i--){
if(S[i]==ch[j]) num++;
if(i+k+1<S.size()&&S[i+k+1]==ch[j]) num--;
s[i].x=max(s[i].x,(dd)(num>0));
}
for(int i=0;i<T.size();i++)
t[y-i-1].x=(dd)(T[i]==ch[j]),sum+=(int)t[y-i-1].x;
FFT(s,1),FFT(t,1);
for(int i=0;i<n;i++) s[i]=s[i]*t[i]; FFT(s,-1);
for(int i=0;i<x;i++) if((int)((dd)s[y+i-1].x+0.5)!=sum) vis[i]=0;
for(int i=0;i<n;i++) s[i].init(),t[i].init();
}
for(int i=0;i<x;i++) ans+=vis[i];
cout<<ans<<endl;
return ;
}
}
int main(){
// freopen("1.in","r",stdin);
Tzh::work();
return 0;
}
CF_528D的更多相关文章
随机推荐
- Ubuntu16.04安装RealSense SR300驱动
原文链接 https://blog.csdn.net/u013401766/article/details/78472285 第一步:CMake 3.14.0 安装 1)下载cmake-3.14.1. ...
- 03 入门 - 安装MVC 5和创建应用程序
目录索引:<ASP.NET MVC 5 高级编程>学习笔记 本篇内容: 1. ASP.NET MVC 5的软件需求 2. 安装ASP.NET MVC 5 1)安装MVC 5开发组件 2)服 ...
- MongoDB学习(配置用户账户和访问控制)
理解admin数据库 安装MongoDB时,会自动创建admin数据库,这是一个特殊的库.有些用户账户角色赋予用户操作多个数据库的权限,而这些用户只能在admin数据库中创建.要创建有权操作所有数据库 ...
- 在docker中初次体验.net core 2.0
.net core的跨平台有了Linux,不能没有docker……网上的系列文章一大推,特别是docker还有了中文官网:https://www.docker-cn.com/ .上面说的很清楚了,这里 ...
- Admin Console 反应慢的相关bug
一个常见问题是在 Admin console 刷新 server 列表时,页面反应慢.从 Admin Server 的 Thread Dump 可以看到 Admin server 到 Managed ...
- pyltp安装踩坑记录
LTP(Language Technology Platform)由哈工大社会计算与信息检索研究中心开发,提供包括中文分词.词性标注.命名实体识别.依存句法分析.语义角色标注等丰富. 高效.精准的自然 ...
- Jmeter简单回顾
之前公众号推文一上手就分享如何测接口, 其实忽略了一些概念性的东西, 今天来给大家拾遗补缺, 做个回顾吧. 一. JMeter介绍 jmeter能做什么,来自官网的解释: Ability to loa ...
- Jmeter、Java当double显示的数字过长时取消科学计数法显示
今日,由于项目需要,使用Jmeter发送查询账户并不保存余额,经过交易后,进行运算后再次比对余额. 实施过程中获取了余额字段并赋值给一个double变量.变量进行运算后再与交易后的账户余额进行比对.这 ...
- Spark RPC框架源码分析(三)Spark心跳机制分析
一.Spark心跳概述 前面两节中介绍了Spark RPC的基本知识,以及深入剖析了Spark RPC中一些源码的实现流程. 具体可以看这里: Spark RPC框架源码分析(二)运行时序 Spark ...
- ORA-12520 TroubleShooting
同事反馈他连接一个新搭建的测试数据库时,报"ORA-12520: TNS: 监听程序无法为请求的服务器类型找到可用的处理程序"错误,在解决他这个问题时,顺便分析.总结一下ORA ...