题目描述

题解:

能看出来的是,每一组数只能改最后一位,所以前$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. hdoj5667 BestCoder Round #80 【费马小定理(膜拜)+矩阵快速幂+快速幂】

    #include<cstdio> #include<string> #include<iostream> #include<vector> #inclu ...

  2. python list生成表达式

    列表生成式即List Comprehensions,是Python内置的非常简单却强大的可以用来创建list的生成式.运用列表生成式,可以写出非常简洁的代码. >>> list(ra ...

  3. web移动端下拉加载数据简单实现

    //下拉加载在移动端会经常使用,有些小伙伴不清楚一些原理下面就简答的介绍一下 //首先需要监听window的滚动事件,下拉其实就是在监听window滚动事件 var pageNum = 1;//分页第 ...

  4. hdu1102 Constructing Roads 基础最小生成树

    //克鲁斯卡尔(最小生成树) #include<cstdio> #include<iostream> #include<algorithm> using names ...

  5. iOS Swift3 用全局“宏”时要注意的问题

    当你需要定义一个APP全局“宏”来调用 UserDefaults.standard里存储的值的时候, 一定要将这个“宏”定义为计算属性,否则你得到的值只会在APP启动的时候计算一次. 示例如下: va ...

  6. 449B

    B. Chtholly's request time limit per test 2 seconds memory limit per test 256 megabytes input standa ...

  7. Jmeter安装说明

    本文主要介绍Jmeter工具的安装 一.安装JDK 1.下载jdk,到官网下载jdk,下载jkd1.8即可,地址:http://www.oracle.com/technetwork/java/java ...

  8. 简单实现人工智能:百度aip+tuling123

    目录结构: app.py # -*- coding: utf-8 -*- # __author: ward # data: 2018/12/21 # @File: app from flask imp ...

  9. 1-3方法的重载(overload)

    之前已经写了一个方法sumInt用来计算两个int类型数字的和,如果要是想计算两个float类型数字的和呢?ok,那就再来写一个sumFloat方法,除此之外,还有long类型,double类型,如果 ...

  10. 162 Find Peak Element 寻找峰值

    峰值元素是指其值大于左右相邻值的元素.给定一个输入数组,其中 num[i] ≠ num[i+1],找到峰值元素并返回其索引.数组可能包含多个峰值,在这种情况下,返回到任何一个峰值所在位置都可以.你可以 ...