URAL1966 Cipher Message 3
题解:
能看出来的是,每一组数只能改最后一位,所以前$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的更多相关文章
- ural Cipher Message
Cipher Message Time Limit:1000MS Memory Limit:65536KB 64bit IO Format:%I64d & %I64u Desc ...
- URAL1996 Cipher Message 3(KMP + FFT)
题目 Source http://acm.timus.ru/problem.aspx?space=1&num=1996 Description Emperor Palpatine has be ...
- URAL 1654 Cipher Message 解题报告
题目链接:http://acm.timus.ru/problem.aspx?space=1&num=1654 题意:简单的理解就是,把一个序列中相邻的且是偶数个相同的字符删除,奇数个的话就只保 ...
- Gym 100285G Cipher Message 3
题意 给\(N,M(N,M \le 250000)\)的两个由8位二进制表示的两个序列,允许改变每个数字的第8位的数值(即0→1,1→0),求改变最少次数使得长为\(M\)的序列为长为\(N\)的连续 ...
- Cipher Message
http://acm.hust.edu.cn/vjudge/contest/view.action?cid=34121#problem/C // File Name: c.cpp // Author: ...
- URAL 1996 Cipher Message 3 (FFT + KMP)
转载请注明出处,谢谢http://blog.csdn.net/ACM_cxlove?viewmode=contents by---cxlove 题意 :给出两个串A , B,每个串是若干个byt ...
- URAL 1996 Cipher Message 3
题目 神题. 记得当初DYF和HZA讲过一个FFT+KMP的题目,一直觉得很神,从来没去做. 没有真正理解FFT的卷积. 首先考虑暴力. 只考虑前7位 KMP 找出所有 B 串可以匹配 A 串的位置. ...
- Ural 1996 Cipher Message 3 (生成函数+FFT)
题面传送门 题目大意:给你两个$01$串$a$和$b$,每$8$个字符为$1$组,每组的最后一个字符可以在$01$之间转换,求$b$成为$a$的一个子串所需的最少转换次数,以及此时是从哪开始匹配的. ...
- URAL 1996. Cipher Message 3(KMP+fft)
传送门 解题思路 因为要完全匹配,所以前七位必须保证相同,那么就可以把前7位提出来做一遍\(kmp\)匹配,最后的答案一定在这些位置里.考虑最后一位,可以把最后一位单独取出来,要计算的是最后一位相同的 ...
随机推荐
- 当打开一个.h或.cpp文件时, Solution Explorer就自动展开文件所在的目录
当打开一个.h或.cpp文件时, Solution Explorer就自动展开文件所在的目录: 如果不想展开: Tools -> Options -&g ...
- python __builtins__ tuple类 (68)
68.'tuple', 转换为元组类型 class tuple(object) | tuple() -> empty tuple | tuple(iterable) -> tuple in ...
- c++,类的对象作为形参时一定会调用复制构造函数吗?
c++,类的对象作为形参时一定会调用复制构造函数吗? 答:如果参数是引用传递,则不会调用任何构造函数:如果是按值传递,则调用复制构造函数,按参数的值构造一个临时对象,这个临时对象仅仅在函数执行是存在, ...
- background-size属性
background-size:属性有 auto:length :百分比 length 如:10px 20px 固定的 或者是写成一个 ,10px 另外一个就默认为 auto; 写成百分比的形式 是 ...
- Distance in Tree CodeForces - 161D
Distance in Tree CodeForces - 161D 题意:给一棵n个结点的树,任意两点之间的距离为1,现在有点u.v,且u与v的最短距离为k,求这样的点对(u,v)的个数((u,v) ...
- BFS Codeforces Round #297 (Div. 2) D. Arthur and Walls
题目传送门 /* 题意:问最少替换'*'为'.',使得'.'连通的都是矩形 BFS:搜索想法很奇妙,先把'.'的入队,然后对于每个'.'八个方向寻找 在2*2的方格里,若只有一个是'*',那么它一定要 ...
- Linux 导出Okular 编辑的pdf批注
1.环境 ubuntu 14.04 LTS Okular Version 0.19.3 Using KDE Development Platform 4.13.3 2.方法 2.1只导出批注,不改变p ...
- git导出代码
1.快速查询 $git archive --format zip --output "./output.zip" master -0 ./output.zip 是生成的文件 mas ...
- 504 Base 7 七进制数
给定一个整数,将其转化为7进制,并以字符串形式输出.示例 1:输入: 100输出: "202" 示例 2:输入: -7输出: "-10"注意: 输入范围是 [- ...
- Hacker的社交礼仪与自我修养【转】
Hacker School是位于纽约的一所特殊的编程“学校”,他们的目标是帮助参与者变成“更好的程序员”,之所以说他们特殊是因为这所“学校”没有老师,没有考试,也不会颁发证书,他们信奉三人行必有我师, ...