[bzoj1892][bzoj2384][bzoj1461][Ceoi2011]Match/字符串的匹配_KMP_树状数组
2384: [Ceoi2011]Match 1892: Match 1461: 字符串的匹配
题目大意:

数据范围:

题解:
很巧妙的一道题呀。
需要对$KMP$算法有很深的理解才行。
首先我们需要发现,要求的这个东西跟字符串匹配有点像。
我们在单个模式串匹配的时候用到的$KMP$算法,合法匹配条件是两个字符完全相同。
但是这个题本质上就是要求子串离散化之后相同。
如果两个串离散化之后完全相同,等价于一个条件,就是每个数前面比它小的个数通通相等。
这是显然的。
所以我们尝试改变$KMP$的匹配模式,并且用树状数组维护长串的这个值。
先假设,所有数字两两不同。
对于要求离散化后的串,每个位置弄一个$f_i$表示这个串中,第$i$个位置前面有多少个比$b_i$小的。
我们把如图红色位置加入树状数组

然后我们查询$i$位置,有多少比$a_i$小的,跟$f_{nxt[i-1]}$相比。
如果相等表示这个位置可以匹配,如果不能,我们就把
$i-nxt_{i-1}$到$i-nxt_{nxt_{i - 1}}$。
这样就可以了。
如果离散化之后不完全相等的话,我们就考虑维护出来$i$前面和$b_i$相等的有多少个,再查就行了。
代码:
#include <bits/stdc++.h>
#define N 1000010
using namespace std;
int tree[N], a[N], b[N], c[N], rk[N], bfr[N], nxt[N], ans[N];
int n, m;
char *p1, *p2, buf[100000];
#define nc() (p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, 100000, stdin), p1 == p2) ? EOF : *p1 ++ )
int rd() {
int x = 0, f = 1;
char c = nc();
while (c < 48) {
if (c == '-')
f = -1;
c = nc();
}
while (c > 47) {
x = (((x << 2) + x) << 1) + (c ^ 48), c = nc();
}
return x * f;
}
inline int lowbit(int x) {
return x & (-x);
}
void update(int x, int val) {
for (int i = x; i <= m; i += lowbit(i))
tree[i] += val;
}
int query(int x) {
int ans = 0;
for (int i = x; i; i -= lowbit(i))
ans += tree[i];
return ans;
}
int main() {
n = rd(), m = rd();
for (int i = 1; i <= n; i ++ )
a[i] = rd(), rk[a[i]] = i;
for (int i = 1; i <= n; i ++ )
bfr[i] = query(rk[i]), update(rk[i], 1);
for (int i = 1; i <= m; i ++ )
b[i] = rd(), c[i] = b[i];
memset(tree, 0, sizeof tree);
// for (int i = 1; i <= n; i ++ )
// printf("%d ", bfr[i]);
// puts("");
for (int i = 2, j = 0; i <= n; i ++ ) {
while (query(rk[i]) != bfr[j + 1]) {
for (int k = i - j; k < i - nxt[j]; k ++ )
update(rk[k], -1);
j = nxt[j];
}
if (query(rk[i]) == bfr[j + 1]) {
update(rk[i], 1);
j ++ ;
}
nxt[i] = j;
}
// for (int i = 1; i <= n; i ++ ) {
// printf("%d ", nxt[i]);
// }
// puts("");
sort(c + 1, c + m + 1);
memset(tree, 0, sizeof tree);
for (int i = 1, j = 0; i <= m; i ++ ) {
// printf("i-> %d\n", i);
b[i] = lower_bound(c + 1, c + m + 1, b[i]) - c;
// printf("%d\n", b[i]);
// printf("%d %d %d\n", j, query(b[i]), bfr[j + 1]);
while (j == n || query(b[i]) != bfr[j + 1]) {
for (int k = i - j; k < i - nxt[j]; k ++ ) {
update(b[k], -1);
}
j = nxt[j];
}
if (query(b[i]) == bfr[j + 1]) {
update(b[i], 1);
j ++ ;
}
if(j == n)
ans[ ++ ans[0]] = i - j + 1;
}
printf("%d\n", ans[0]);
for (int i = 1; i < ans[0]; i ++ )
printf("%d ",ans[i]);
if(ans[0])
printf("%d\n", ans[ans[0]]);
return 0;
}
小结:好题啊,这个题真的不好想,我看题解都看了半天.......
[bzoj1892][bzoj2384][bzoj1461][Ceoi2011]Match/字符串的匹配_KMP_树状数组的更多相关文章
- 【poj 3167】Cow Patterns(字符串--KMP匹配+数据结构--树状数组)
题意:给2个数字序列 a 和 b ,问按从小到达排序后,a中的哪些子串与b的名次匹配. a 的长度 N≤100,000,b的长度 M≤25,000,数字的大小 K≤25. 解法:[思考]1.X 暴力. ...
- BZOJ_1264_[AHOI2006]基因匹配Match_树状数组
BZOJ_1264_[AHOI2006]基因匹配Match_树状数组 Description 基因匹配(match) 卡卡昨天晚上做梦梦见他和可可来到了另外一个星球,这个星球上生物的DNA序列由无数种 ...
- 洛谷P4303 [AHOI2006]基因匹配(树状数组)
传送门 我已经连这种傻逼题都不会了orz 正常的dp是$O(n^2)$的,枚举第一个数组的$j$,然后第二个数组的$k$,如果相等,则$dp[i]=dp[j]+1$,否则$dp[i]=dp[j]$ 然 ...
- 【bzoj2384】[Ceoi2011]Match 特殊匹配条件的KMP+树状数组
题目描述 给出两个长度分别为n.m的序列A.B,求出B的所有长度为n的连续子序列(子串),满足:序列中第i小的数在序列的Ai位置. 输入 第一行包含两个整数n, m (2≤n≤m≤1000000). ...
- 【BZOJ2384】[Ceoi2011]Match KMP
[BZOJ2384][Ceoi2011]Match Description 作为新一轮广告大战的一部分,格丁尼亚的一家大公司准备在城市的某处设置公司的标志(logo).公司经理决定用一些整栋的建筑来构 ...
- bzoj 1264 [AHOI2006]基因匹配Match(DP+树状数组)
1264: [AHOI2006]基因匹配Match Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 793 Solved: 503[Submit][S ...
- 【BZOJ1264】[AHOI2006]基因匹配Match DP+树状数组
[BZOJ1264][AHOI2006]基因匹配Match Description 基因匹配(match) 卡卡昨天晚上做梦梦见他和可可来到了另外一个星球,这个星球上生物的DNA序列由无数种碱基排列而 ...
- bzoj1264 [AHOI2006]基因匹配Match 树状数组+lcs
1264: [AHOI2006]基因匹配Match Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1255 Solved: 835[Submit][ ...
- BZOJ 1264: [AHOI2006]基因匹配Match 树状数组+DP
1264: [AHOI2006]基因匹配Match Description 基因匹配(match) 卡卡昨天晚上做梦梦见他和可可来到了另外一个星球,这个星球上生物的DNA序列由无数种碱基排列而成(地球 ...
随机推荐
- Laravel 多态关联中利用关联表相关字段进行排序的问题
1 目标 1.1 在 Laravel 项目的开发中,多态的需求很常见,按多态关联进行排序的需求也是必须的. 1.2 请想像,我们有一个需求,荣誉栏目多态关联一个档案模型,要求在荣誉中按档案的推荐时间进 ...
- CF1230 E. Kamil and Making a Stream gcd+暴力
比赛的时候TLE,第二天发现合并方向合并错了~ 改了一下顺序就切了~ 又掉分了,好难过QAQ...... Code: #include <bits/stdc++.h> #define N ...
- spring boot+idea实现程序热部署
pring为开发者提供了一个名为spring-boot-devtools的模块来使Spring Boot应用支持热部署,提高开发者的开发效率,无需手动重启Spring Boot应用. devtools ...
- javascript原型继承
在传统的基于Class的语言如Java.C++中,继承的本质是扩展一个已有的Class,并生成新的Subclass. 由于这类语言严格区分类和实例,继承实际上是类型的扩展.但是,JavaScript由 ...
- 函数柯里化与偏函数+bind
简单理解: 1,函数柯里化就是把多参数函数分解为多return的单参数函数: 举个例子(伪代码): function func (a, b, c){ return } 柯里化为 function fu ...
- iOS 8 - Can't Install Enterprise App
http://stackoverflow.com/questions/25733299/ios-8-cant-install-enterprise-app 版权声明:本文为博主原创文章,未经博主允许不 ...
- LeetCode 10. 正则表达式匹配(Regular Expression Matching)
题目描述 给定一个字符串 (s) 和一个字符模式 (p).实现支持 '.' 和 '*' 的正则表达式匹配. '.' 匹配任意单个字符. '*' 匹配零个或多个前面的元素. 匹配应该覆盖整个字符串 (s ...
- LeetCode 124. 二叉树中的最大路径和(Binary Tree Maximum Path Sum)
题目描述 给定一个非空二叉树,返回其最大路径和. 本题中,路径被定义为一条从树中任意节点出发,达到任意节点的序列.该路径至少包含一个节点,且不一定经过根节点. 示例 1: 输入: [1,2,3] 1 ...
- postgresql数据库的 to_date 和 to_timestamp 将 字符串转换为时间格式
数据库中:字符串 转换为 时间格式 二者区别: to_data 转换为 普通的时间格式 to_timestamp 转换可为 时间戳格式出错场景: 比较同一天 日期大小的时候,很容易出错 ...
- Qt子窗口QMidSubwindow全屏出现的问题总结
我的需求:想全屏一个子窗口QMidSubwindow,禁止显示最大化最小化和关闭按钮. 我开始尝试的是网上介绍的方法,把结果展现给大家一下,最后再总结: 方法1:QMidSubwindow直接调用sh ...