BZOJ 4503: 两个串
来源:互联网 发布:网络时时彩案件判刑 编辑:程序博客网 时间:2024/06/05 10:21
Description
兔子们在玩两个串的游戏。给定两个字符串S和T,兔子们想知道T在S中出现了几次,
分别在哪些位置出现。注意T中可能有“?”字符,这个字符可以匹配任何字符。
Input
两行两个字符串,分别代表S和T
Output
第一行一个正整数k,表示T在S中出现了几次
接下来k行正整数,分别代表T每次在S中出现的开始位置。按照从小到大的顺序输出,S下标从0开始。
Sample Input
bbabaababaaaaabaaaaaaaabaaabbbabaaabbabaabbbbabbbbbbabbaabbbababababbbbbbaaabaaabbbbbaabbbaabbbbabab
a?aba?abba
Sample Output
0
HINT
S 长度不超过 10^5, T 长度不会超过 S。 S 中只包含小写字母, T中只包含小写字母和“?”
分析
这题居然不是后缀数组之类的乱搞也是很迷。。用fft来解决就更迷了。。
对于这道题如果没有通配符,我们构造一个函数
这样如果s1和s2相等当且仅当f[x]=0
但是这道题有通配符,所以我们要把是通配符的位置s2[i]=0
然后变换一下公式可得:
这样有通配符的位置相应的函数值就会为0,表示可以匹配一切
但是这样没法做,于是将s2数组拧一拧,翻转过来,发现就是一个卷积之类的了,可用FFT。
代码
#include <bits/stdc++.h>#define N 300005#define pi acos(-1)#define ll long longusing namespace std;typedef complex<double> E;int L,n;int R[N];E c[N],d[N],e[N],f[N];int a[N],b[N];char s1[N],s2[N];void fft(E *a,int f){ for (int i = 0; i < n; i++) if (i < R[i]) swap(a[i],a[R[i]]); for (int i = 1; i < n; i <<= 1) { E wn(cos(pi / i), f * sin(pi / i)); for (int j = 0; j < n; j += (i << 1)) { E w(1,0); for (int k = 0; k < i; k++, w *= wn) { E x = a[j + k], y = w * a[j + k + i]; a[j + k] = x + y; a[j + k + i] = x - y; } } } if (f == -1) for (int i = 0; i < n; i++) a[i] /= n;}int main(){ scanf("%s",s1 + 1); scanf("%s",s2 + 1); int len1 = strlen(s1 + 1), len2 = strlen(s2 + 1); for (int i = 1; i <= len1; i++) a[i] = s1[i] - 'a' + 1; for (int i = 1; i <= len2; i++) b[len2 - i + 1] = s2[i] == '?' ? 0 : s2[i] - 'a' + 1; for (n = 1; n <= len1 * 2; n <<= 1, L++); for (int i = 0; i < N; i++) R[i] = (R[i >> 1] >> 1) | ((i & 1) << (L - 1)); int w = 0; for (int i = 1; i <= len2; i++) w += b[i] * b[i] * b[i]; for (int i = 1; i <= len1; i++) c[i] = a[i] * a[i]; for (int i = 1; i <= len2; i++) d[i] = b[i]; for (int i = 1;i <= len1; i++) e[i] = a[i]; for (int i = 1; i <= len2; i++) f[i] = b[i] * b[i] * 2; fft(c,1); fft(d,1); fft(e,1); fft(f,1); for (int i = 0; i < n; i++) c[i] = c[i] * d[i], e[i] = e[i] * f[i]; fft(c,-1); fft(e,-1); for (int i = 0; i < N; i++) a[i] = (int)(c[i].real() + 0.1) - (int)(e[i].real() + 0.1); int tot = 0; for (int i = len2 + 1; i <= len1 + 1; i++) if (a[i] + w == 0) tot++; cout<<tot<<endl; for (int i = len2 + 1; i <= len1 + 1; i++) if (a[i] + w == 0) printf("%d\n",i - len2 - 1);}
0 0
- 【BZOJ 4503】两个串
- BZOJ 4503 两个串
- BZOJ 4503: 两个串
- bzoj 4503 两个串
- bzoj 4503: 两个串 fft
- bzoj 4503: 两个串 fft
- bzoj 4503: 两个串 (FFT+DP)
- BZOJ 4503: 两个串 FFT 通配符匹配
- [BZOJ]4503 两个串:我的第一次FFT尝试
- 4503: 两个串 FFT
- bzoj 2740: 串
- bzoj-3676 回文串
- [BZOJ 2946]公共串
- BZOJ 3676 回文串
- [BZOJ ]
- BZOJ****-****
- BZOJ
- BZOJ
- Leetcode题解
- GoogLeNet( Going deeper with convolutions)解读
- 标签的属性值,可以自己任意定义如
- java基础(2)
- Exception in thread "main" org.openqa.selenium.remote.UnreachableBrowserException
- BZOJ 4503: 两个串
- JAVA笔记复习2
- Vue DevTools可使用修正方法
- Linux Kernel 学习笔记3:设备编号
- How to Check or Verify PC Motherboard BIOS SLIC Version is SLP OA 2.0 or 2.1 for OEM Activation
- 传输协议
- MyEclipse常用插件使用教程
- ScrollView中嵌入Listview,当item高度不一样的时候,item展示不全问题
- 算法链表合并