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_树状数组的更多相关文章

  1. 【poj 3167】Cow Patterns(字符串--KMP匹配+数据结构--树状数组)

    题意:给2个数字序列 a 和 b ,问按从小到达排序后,a中的哪些子串与b的名次匹配. a 的长度 N≤100,000,b的长度 M≤25,000,数字的大小 K≤25. 解法:[思考]1.X 暴力. ...

  2. BZOJ_1264_[AHOI2006]基因匹配Match_树状数组

    BZOJ_1264_[AHOI2006]基因匹配Match_树状数组 Description 基因匹配(match) 卡卡昨天晚上做梦梦见他和可可来到了另外一个星球,这个星球上生物的DNA序列由无数种 ...

  3. 洛谷P4303 [AHOI2006]基因匹配(树状数组)

    传送门 我已经连这种傻逼题都不会了orz 正常的dp是$O(n^2)$的,枚举第一个数组的$j$,然后第二个数组的$k$,如果相等,则$dp[i]=dp[j]+1$,否则$dp[i]=dp[j]$ 然 ...

  4. 【bzoj2384】[Ceoi2011]Match 特殊匹配条件的KMP+树状数组

    题目描述 给出两个长度分别为n.m的序列A.B,求出B的所有长度为n的连续子序列(子串),满足:序列中第i小的数在序列的Ai位置. 输入 第一行包含两个整数n, m (2≤n≤m≤1000000).  ...

  5. 【BZOJ2384】[Ceoi2011]Match KMP

    [BZOJ2384][Ceoi2011]Match Description 作为新一轮广告大战的一部分,格丁尼亚的一家大公司准备在城市的某处设置公司的标志(logo).公司经理决定用一些整栋的建筑来构 ...

  6. bzoj 1264 [AHOI2006]基因匹配Match(DP+树状数组)

    1264: [AHOI2006]基因匹配Match Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 793  Solved: 503[Submit][S ...

  7. 【BZOJ1264】[AHOI2006]基因匹配Match DP+树状数组

    [BZOJ1264][AHOI2006]基因匹配Match Description 基因匹配(match) 卡卡昨天晚上做梦梦见他和可可来到了另外一个星球,这个星球上生物的DNA序列由无数种碱基排列而 ...

  8. bzoj1264 [AHOI2006]基因匹配Match 树状数组+lcs

    1264: [AHOI2006]基因匹配Match Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1255  Solved: 835[Submit][ ...

  9. BZOJ 1264: [AHOI2006]基因匹配Match 树状数组+DP

    1264: [AHOI2006]基因匹配Match Description 基因匹配(match) 卡卡昨天晚上做梦梦见他和可可来到了另外一个星球,这个星球上生物的DNA序列由无数种碱基排列而成(地球 ...

随机推荐

  1. springboot的@Configuration文件读取static静态文件

    错误 正确

  2. .NET大文件分片上传

    需求:项目要支持大文件上传功能,经过讨论,初步将文件上传大小控制在500M内,因此自己需要在项目中进行文件上传部分的调整和配置,自己将大小都以501M来进行限制. 第一步: 前端修改 由于项目使用的是 ...

  3. 小米 oj 发奖励(思维)

     发奖励 序号:#75难度:有挑战时间限制:1000ms内存限制:10M 描述 小明老师准备给一些得到小红花的小朋友发糖果做为奖励. 假设有n个小朋友,每个小朋友拥有的小红花为m(n)个,他让这n个小 ...

  4. ege图形库之动画排序

    老师布置了一个学习ege图形库来做动画排序的小动画程序,这是我自己做的效果.由于个人水平有限,可能代码有些地方可以改进.不足之处敬请指出. 注:要运行该代码需要正确配置,连接好ege图形库的头文件,做 ...

  5. 【csp模拟赛5】加减法--宽搜维护联通快

    题目大意: 一开始想用并查集,发现很难维护联通块的代表元素,所以用了宽搜,开数组会炸,所以开一个优先队列维护,每扫完一个联通块,统计答案,清空优先队列,!!千万记住注意数组的大小!!! 代码: #in ...

  6. 制作自己的livecd

    把现有ubuntu系统打包,迁移到另外一台硬件一模一样的机器上. https://zhuanlan.zhihu.com/p/51827233 https://blog.csdn.net/vah101/ ...

  7. 栈(Java实现)

    栈是最基本的数据结构之一,其特点是先进后出. 1.基于数组的可动态调节大小的栈 public class ResizingArrayStack<Item> { private Item[] ...

  8. centos 安装mariadb 替代mysql

    yum install mariadb-server mariadb systemctl start mariadbmysql -uroot -p默认密码mysql -uroot -pmysql_se ...

  9. 状压dp,松鼠从起点出发,拿到所有坚果,然后返回起点,求最短时间。

    UVA10944 松鼠从起点出发,拿到所有坚果,然后返回起点,求最短时间. #include<iostream> #include<cstdio> #include<al ...

  10. PPT:很多文字如何排版?(PPT如何美化?)

    1.加入图标.形状(或者加入图片和色块搭配) 2.提取关键词作为标题(这个很重要) 下面看下示例: 这个是没修改前 修改后: 相信大家都会喜欢第二张吧 再看一个: 最近在学做PPT,有点上瘾了,哈哈哈 ...