弱省互测#2 t3
题意
给出\(n\)个01字节和\(m\)个01字节,要求用后者去匹配前者,两个串能匹配当且仅当除了每个字节末位不同,其他位都要相同。问匹配后者至少有多少个末位不同。(\(1 \le m \le n \le 2.5 \times 10^5\))
分析
首先我们可以用kmp计算出能匹配的位置,然后单独考虑末位不同的情况。
题解
我们将末尾的位提取出来,则考虑\(n\)个\(01\)位和\(m\)个\(01\)位。对于模板串的\(01\)位,我们需要计算以这个位置结束与匹配串位相同的数目,发现其实我们将匹配串反转,然后就是卷积!于是我们就可以用fft做了。
#include <bits/stdc++.h>
using namespace std;
const double pi=acos(-1);
const int N=2.5e5+10, nS=N*8, Lim=600006;
int lenn, lenm, lena, lenb, got[Lim], rev[Lim];
char s1[nS], s2[nS], sa[nS], sb[nS];
vector<int> pos;
void getkmp() {
static int p[nS];
memset(p, -1, sizeof(int)*lena);
int j=-1;
for(int i=1; i<lena; ++i) {
while(j!=-1 && sa[j+1]!=sa[i]) j=p[j];
if(sa[j+1]==sa[i]) ++j;
p[i]=j;
}
j=-1;
for(int i=0; i<lena; ++i) {
while(j!=-1 && sb[j+1]!=sa[i]) j=p[j];
if(sb[j+1]==sa[i]) ++j;
if(j==lenb-1) {
int t=i-j;
if(t%7==0) {
pos.push_back(t/7);
}
}
}
}
struct icp {
double r, i;
icp(double _r=0, double _i=0) : r(_r), i(_i) { }
};
icp operator + (const icp &a, const icp &b) { return icp(a.r+b.r, a.i+b.i); }
icp operator - (const icp &a, const icp &b) { return icp(a.r-b.r, a.i-b.i); }
icp operator * (const icp &a, const icp &b) { return icp(a.r*b.r-a.i*b.i, a.r*b.i+a.i*b.r); }
int getlen(int n) {
int len=1, bl=-1;
for(; len<n; len<<=1, ++bl);
for(int i=0; i<len; ++i) {
rev[i]=(rev[i>>1]>>1)|((i&1)<<bl);
}
return len;
}
void fft(icp *a, int n, int flag) {
for(int i=0; i<n; ++i) {
if(rev[i]<i) {
swap(a[rev[i]], a[i]);
}
}
for(int m=2; m<=n; m<<=1) {
icp wn(cos(pi*2/m), sin(pi*2/m)*flag);
for(int i=0, mid=m>>1; i<n; i+=m) {
icp w(1);
for(int j=0; j<mid; ++j) {
icp u=a[i+j], v=a[i+j+mid]*w;
a[i+j]=u+v;
a[i+j+mid]=u-v;
w=w*wn;
}
}
}
if(flag==-1) {
for(int i=0; i<n; ++i) {
a[i].r/=n;
}
}
}
void dofft(char *A, char *B, int *C, int n, int m) {
#define CLR(a) for(int i=0; i<len; ++i) a[i].r=a[i].i=0;
static icp a[Lim], b[Lim], c[Lim];
int len=getlen(n+m-1);
CLR(a);
CLR(b);
CLR(c);
for(int i=0; i<n; ++i)
a[i].r=A[i]-'0';
for(int i=0; i<m; ++i)
b[i].r=B[i]-'0';
fft(a, len, 1);
fft(b, len, 1);
for(int i=0; i<len; ++i)
c[i]=a[i]*b[i];
fft(c, len, -1);
for(int i=0; i<len; ++i)
C[i]+=c[i].r+0.5;
}
void getfft() {
lena=lenb=0;
for(int i=7; i<lenn; i+=8)
sa[lena++]=s1[i];
for(int i=lenm-1; i>=0; i-=8)
sb[lenb++]=s2[i];
dofft(sa, sb, got, lena, lenb);
for(int i=0; i<lena; ++i)
sa[i]=sa[i]=='0'?'1':'0';
for(int i=0; i<lenb; ++i)
sb[i]=sb[i]=='0'?'1':'0';
dofft(sa, sb, got, lena, lenb);
}
int main() {
int n, m;
scanf("%d%d", &n, &m);
lenn=n*8, lenm=m*8;
char *it;
it=s1;
for(int i=0; i<n; ++i) {
scanf("%s", it);
for(; *it; ++it) {
if(*(it+1)) {
sa[lena++]=*it;
}
}
}
it=s2;
for(int i=0; i<m; ++i) {
scanf("%s", it);
for(; *it; ++it) {
if(*(it+1)) {
sb[lenb++]=*it;
}
}
}
getkmp();
getfft();
if(!pos.size()) {
puts("No");
return 0;
}
int ans1=n, ans2=~0u>>1;
for(int i=0, len=pos.size(); i<len; ++i) {
int p=pos[i], t=(m-got[p+m-1]);
if(ans2==t) {
ans1=min(p, ans1);
}
else if(ans2>t) {
ans2=t;
ans1=p;
}
}
printf("Yes\n%d %d\n", ans2, ans1+1);
return 0;
}
弱省互测#2 t3的更多相关文章
- 弱省互测#1 t3
题意 给出一棵n个点的树,求包含1号点的第k小的连通块权值和.(\(n<=10^5\)) 分析 k小一般考虑堆... 题解 堆中关键字为\(s(x)+min(a)\),其中\(s(x)\)表示\ ...
- 弱省互测#0 t3
Case 1 题意 要求给出下面代码的答案然后构造输入. 给一个图, n 个点 m 条边 q 次询问,输出所有点对之间最大权值最小的路径. 题解 把每一个询问的输出看成一条边,建一棵最小生成树. Ca ...
- 【CH 弱省互测 Round #1 】OVOO(可持久化可并堆)
Description 给定一颗 \(n\) 个点的树,带边权. 你可以选出一个包含 \(1\) 顶点的连通块,连通块的权值为连接块内这些点的边权和. 求一种选法,使得这个选法的权值是所有选法中第 \ ...
- 弱省互测#2 t2
题意 给两个树,大小分别为n和m,现在两棵树各选一些点(包括1),使得这棵树以1号点为根同构(同构就是每个点的孩子数目相同),求最大的同构树.(n, m<=500) 分析 我们从两棵树中各取出一 ...
- 弱省互测#0 t2
题意 给定两个字符串 A 和 B,求下面四个问题的答案: 1.在 A 的子串中,不是 B 的子串的字符串的数量. 2.在 A 的子串中,不是 B 的子序列的字符串的数量. 3.在 A 的子序列中,不是 ...
- 弱省互测#0 t1
题意 给一个\(N \times M\)的01网格,1不能走,从起点\((1, 1)\)走到\((N, M)\),每次只能向下或向右走一格,问两条不相交的路径的方案数.(n, m<=1000) ...
- 【loj2461】【2018集训队互测Day 1】完美的队列
#2461. 「2018 集训队互测 Day 1」完美的队列 传送门: https://loj.ac/problem/2461 题解: 直接做可能一次操作加入队列同时会弹出很多数字,无法维护:一个操作 ...
- 【2018集训队互测】【XSY3372】取石子
题目来源:2018集训队互测 Round17 T2 题意: 题解: 显然我是不可能想出来的……但是觉得这题题解太神了就来搬(chao)一下……Orzpyz! 显然不会无解…… 为了方便计算石子个数,在 ...
- 洛谷 P4463 - [集训队互测 2012] calc(多项式)
题面传送门 & 加强版题面传送门 竟然能独立做出 jxd 互测的题(及其加强版),震撼震撼(((故写题解以祭之 首先由于 \(a_1,a_2,\cdots,a_n\) 互不相同,故可以考虑求出 ...
随机推荐
- docker版wordpress
拉取wordpress镜像 docker pull wordpress:latest 创建mysql 容器docker run --name wordpress-mysql -e MYSQL_ROOT ...
- MAC实用的小工具
一.XtraFinder(右键菜单扩展) http://www.xuebuyuan.com/173454.html http://www.mamicode.com/info-detail-111618 ...
- 按钮button的css样式(扁平化底色)
.button { background-color: #ff0000; /* Green */ border: none; color: white; font-family:Arial; padd ...
- magento后台使用POST表单时,要使用必要参数form_key才能正常通讯
<form action="<?php echo $this->getSaveUrl() ?>" method="POST" encty ...
- c#程序如何从海康ipserver查询获取DVR设备ip地址
海康威视提供的ipserver可以记录DVR设备的IP地址,并提供一个7071的端口给客户端查询用,我们在c#程序中可以用海康SDK包中的NET_DVR_GetDVRIPByResolveSvr函数来 ...
- 基于SSM的租赁管理系统0.2_20161225_开发环境
项目环境搭建 1. 开发环境 Sybase PowerDesigner 15.1.0 + MySQL 5.7.15 + Navicat 11.0.9 + eclipse EE Mars 2.0 + F ...
- 模拟搭建Web项目的真实运行环境(七)
下面这个是mongo驱动的小案例,里面也有涉及到一点redis的操作 https://github.com/SuperRocky/MyMongoDriver 接下来通过几张图片主要介绍一下每个文件的具 ...
- js与jq对数组的操作
一.数组处理 1.数组的创建 var arrayObj = new Array(); //创建一个数组 var arrayObj = new Array([size]); //创建一个数组并指定长 ...
- C++开始前篇,深入编译链接(补充2)
在开始链接之前,我们先了解几个概念: 一>符号的概念. 我们知道,链接的最重要的是"对符号的重定位",而且上面提到了符号表,那什么是符号呢,在链接中,我们将函数和变量统称为符 ...
- NetBeans无法使用编码GBK安全打开文件
刚才使用NetBeans打开php文件时,提示:NetBeans无法使用编码GBK安全地打开该路径下的文件. 找到了解决方案. 原文地址:http://qdjinxin.iteye.com/blog/ ...