题目描述

题解:

能看出来的是,每一组数只能改最后一位,所以前$7$位动不了。

所以$KMP$跑一跑。

重点在于最后一位怎么搞。

如果$KMP$跑完了还没找到合适的位置,直接$puts("No")$就好了。

剩下个匹配问题。

(要不是数据范围拦着我我都想建图跑费用流了)

这个匹配可以用$FFT$求。

$FFT$不是求$\sum(a[i]*b[j-i])$的吗?怎么求字符串匹配?

其实我们只想要最后多项式中某一位上的值,所以$b[j-i]$和$b[i]$没有区别,反转数组就好了。

我们要求的是需要多少个$0$变成$1$,以及多少个$1$变成$0$。

所以$FFT$求两遍卷积,第一次$a[i]=[s1[i]==0],b[i]=[s2[i]==1]$,注意这里的$b$还没有反转。

第二次反过来就可以了。

代码:

#include<cmath>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef unsigned long long ull;
const int N = ;
const ull seed = ;
const double Pi = acos(-1.0);
template<typename T>
inline void read(T&x)
{
T f = ,c = ;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){c=c*+ch-'';ch=getchar();}
x = f*c;
}
struct cp
{
double x,y;
cp(){}
cp(double x,double y):x(x),y(y){}
cp operator + (const cp&a)const{return cp(x+a.x,y+a.y);}
cp operator - (const cp&a)const{return cp(x-a.x,y-a.y);}
cp operator * (const cp&a)const{return cp(x*a.x-y*a.y,x*a.y+y*a.x);}
};
int to[*N],lim=,l;
void fft(cp *a,int len,int k)
{
for(int i=;i<len;i++)
if(i<to[i])swap(a[i],a[to[i]]);
for(int i=;i<len;i<<=)
{
cp w0(cos(Pi/i),k*sin(Pi/i));
for(int j=;j<len;j+=(i<<))
{
cp w(,);
for(int o=;o<i;o++,w=w*w0)
{
cp w1 = a[j+o],w2 = a[j+o+i]*w;
a[j+o] = w1+w2;
a[j+o+i] = w1-w2;
}
}
}
if(k==-)
for(int i=;i<len;i++)
a[i].x/=len;
}
void rd(int n,int *a1,int *a2)
{
for(int x,i=;i<=n;i++)
{
read(x);
a1[i] = x%;
a2[i] = x/;
}
}
int n,m,a[][N],b[][N];
bool vis[N];
int ans[N];
cp c1[*N],c2[*N],c3[*N];
int nxt[N];
void get_nxt(int *a,int len)
{
int i = ,j = ;
while(i<=len)
{
if(!j||a[i]==a[j])
{
i++,j++;
nxt[i] = j;
}else j = nxt[j];
}
}
int kmp(int *a,int la,int *b,int lb)
{
int ret = ;
int i = ,j = ;
while(i<=la)
{
if(a[i]==b[j]||j==)
{
i++,j++;
if(j==lb+)
{
vis[i-]=;
j = nxt[j];
ret = ;
}
}else j=nxt[j];
}
return ret;
}
void work()
{
fft(c1,lim,),fft(c2,lim,);
for(int i=;i<lim;i++)c3[i]=c1[i]*c2[i];
fft(c3,lim,-);
}
void init()
{
memset(a,,sizeof(a));
memset(b,,sizeof(b));
memset(ans,,sizeof(ans));
memset(vis,,sizeof(vis));
lim=,l=;
}
int main()
{
// freopen("tt.in","r",stdin);
while(scanf("%d%d",&n,&m)>)
{
init();
rd(n,a[],a[]),rd(m,b[],b[]);
get_nxt(b[],m);
if(!kmp(a[],n,b[],m))
{
puts("No");
continue;
}
puts("Yes");
while(lim<*(n+m))lim<<=,l++;
for(int i=;i<lim;i++)to[i]=((to[i>>]>>)|((i&)<<(l-)));
for(int i=;i<lim;i++)c1[i]=c2[i]=cp(,);
for(int i=;i<n;i++)c1[i].x=(a[][i+]==);
for(int i=;i<m;i++)c2[i].x=(b[][m-i]==);
work();
for(int i=;i<n;i++)ans[i+]=(int)(c3[i].x+0.5);
for(int i=;i<lim;i++)c1[i]=c2[i]=cp(,);
for(int i=;i<n;i++)c1[i].x=(a[][i+]==);
for(int i=;i<m;i++)c2[i].x=(b[][m-i]==);
work();
for(int i=;i<n;i++)ans[i+]+=(int)(c3[i].x+0.5);
int mn = N,pos=;
for(int i=m;i<=n;i++)
if(vis[i]&&ans[i]<mn)
{
mn = ans[i];
pos = i-m+;
}
printf("%d %d\n",mn,pos);
}
return ;
}

URAL1966 Cipher Message 3的更多相关文章

  1. ural Cipher Message

    Cipher Message Time Limit:1000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u Desc ...

  2. URAL1996 Cipher Message 3(KMP + FFT)

    题目 Source http://acm.timus.ru/problem.aspx?space=1&num=1996 Description Emperor Palpatine has be ...

  3. URAL 1654 Cipher Message 解题报告

    题目链接:http://acm.timus.ru/problem.aspx?space=1&num=1654 题意:简单的理解就是,把一个序列中相邻的且是偶数个相同的字符删除,奇数个的话就只保 ...

  4. Gym 100285G Cipher Message 3

    题意 给\(N,M(N,M \le 250000)\)的两个由8位二进制表示的两个序列,允许改变每个数字的第8位的数值(即0→1,1→0),求改变最少次数使得长为\(M\)的序列为长为\(N\)的连续 ...

  5. Cipher Message

    http://acm.hust.edu.cn/vjudge/contest/view.action?cid=34121#problem/C // File Name: c.cpp // Author: ...

  6. URAL 1996 Cipher Message 3 (FFT + KMP)

    转载请注明出处,谢谢http://blog.csdn.net/ACM_cxlove?viewmode=contents    by---cxlove 题意 :给出两个串A , B,每个串是若干个byt ...

  7. URAL 1996 Cipher Message 3

    题目 神题. 记得当初DYF和HZA讲过一个FFT+KMP的题目,一直觉得很神,从来没去做. 没有真正理解FFT的卷积. 首先考虑暴力. 只考虑前7位 KMP 找出所有 B 串可以匹配 A 串的位置. ...

  8. Ural 1996 Cipher Message 3 (生成函数+FFT)

    题面传送门 题目大意:给你两个$01$串$a$和$b$,每$8$个字符为$1$组,每组的最后一个字符可以在$01$之间转换,求$b$成为$a$的一个子串所需的最少转换次数,以及此时是从哪开始匹配的. ...

  9. URAL 1996. Cipher Message 3(KMP+fft)

    传送门 解题思路 因为要完全匹配,所以前七位必须保证相同,那么就可以把前7位提出来做一遍\(kmp\)匹配,最后的答案一定在这些位置里.考虑最后一位,可以把最后一位单独取出来,要计算的是最后一位相同的 ...

随机推荐

  1. 当打开一个.h或.cpp文件时, Solution Explorer就自动展开文件所在的目录

    当打开一个.h或.cpp文件时,  Solution Explorer就自动展开文件所在的目录: 如果不想展开: Tools           -> Options           -&g ...

  2. python __builtins__ tuple类 (68)

    68.'tuple', 转换为元组类型 class tuple(object) | tuple() -> empty tuple | tuple(iterable) -> tuple in ...

  3. c++,类的对象作为形参时一定会调用复制构造函数吗?

    c++,类的对象作为形参时一定会调用复制构造函数吗? 答:如果参数是引用传递,则不会调用任何构造函数:如果是按值传递,则调用复制构造函数,按参数的值构造一个临时对象,这个临时对象仅仅在函数执行是存在, ...

  4. background-size属性

    background-size:属性有 auto:length :百分比 length 如:10px 20px 固定的 或者是写成一个 ,10px  另外一个就默认为 auto; 写成百分比的形式 是 ...

  5. Distance in Tree CodeForces - 161D

    Distance in Tree CodeForces - 161D 题意:给一棵n个结点的树,任意两点之间的距离为1,现在有点u.v,且u与v的最短距离为k,求这样的点对(u,v)的个数((u,v) ...

  6. BFS Codeforces Round #297 (Div. 2) D. Arthur and Walls

    题目传送门 /* 题意:问最少替换'*'为'.',使得'.'连通的都是矩形 BFS:搜索想法很奇妙,先把'.'的入队,然后对于每个'.'八个方向寻找 在2*2的方格里,若只有一个是'*',那么它一定要 ...

  7. Linux 导出Okular 编辑的pdf批注

    1.环境 ubuntu 14.04 LTS Okular Version 0.19.3 Using KDE Development Platform 4.13.3 2.方法 2.1只导出批注,不改变p ...

  8. git导出代码

    1.快速查询 $git archive --format zip --output "./output.zip" master -0 ./output.zip 是生成的文件 mas ...

  9. 504 Base 7 七进制数

    给定一个整数,将其转化为7进制,并以字符串形式输出.示例 1:输入: 100输出: "202" 示例 2:输入: -7输出: "-10"注意: 输入范围是 [- ...

  10. Hacker的社交礼仪与自我修养【转】

    Hacker School是位于纽约的一所特殊的编程“学校”,他们的目标是帮助参与者变成“更好的程序员”,之所以说他们特殊是因为这所“学校”没有老师,没有考试,也不会颁发证书,他们信奉三人行必有我师, ...