B君的第二题



分析

题目就是让你求文本框与s匹配到了哪个位置。

考场70分

一看就是裸的kmp,直接打上去。

const int MAXN=1e5+8;
char s[MAXN],t[MAXN];
int n,m;
int nx[MAXN];
int f[MAXN],len; int main()
{
freopen("xining.in","r",stdin);
freopen("xining.out","w",stdout);
scanf("%s",s+1);
n=strlen(s+1);
scanf("%s",t+1);
m=strlen(t+1); for(int i=2;i<=n;++i)
{
int p=nx[i-1];
while(p&&s[p+1]!=s[i])
p=nx[p];
if(s[p+1]==s[i])
++p;
nx[i]=p;
} printf("%d\n",n);
for(int i=1;i<=m;++i)
{
if(t[i]=='-')
{
--len;
if(len<0)
len=0;
printf("%d\n",n-f[len]);
continue;
}
// cerr<<"pro "<<i<<endl;
int p=f[len];
// cerr<<" p="<<p<<endl;
while(p&&s[p+1]!=t[i])
p=nx[p];
if(s[p+1]==t[i])
++p;
f[++len]=p;
// cerr<<" p="<<p<<endl;
printf("%d\n",n-p);
}
// fclose(stdin);
// fclose(stdout);
return 0;
}

结果TLE了。实际上时间复杂度\(O(n^2)\)

标解

这道题能卡kmp,是因为往回走后重新匹配跳next时复杂度没了保证。

退格操作可以让匹配长度剧增,这与kmp要满足长度增加量最多为文本串的长度的性质不符,所以要被卡。

B君:可以前n/2都是a,然后后面一个-一个b交替出现,就可以把kmp卡成n方的了。

但是AC自动机中的一个构建优化是把不存在的儿子补齐,所以考虑AC自动机。

然后因为AC自动机是在Trie图上建的,距离不能直接求,所以要预处理。

时间复杂度\(O(n)\)

const int MAXN=1e5+7;

vector<int>a[MAXN];
int d[MAXN]; const int CHARSET=26;
struct Trie
{
int sz;
int to[MAXN][CHARSET];
int nx[MAXN];
int fa[MAXN];
int v[MAXN]; il void ins(rg char*s)
{
rg int p=0;
for(;*s;++s)
{
rg int c=*s-'a';
if(to[p][c]==0)
{
to[p][c]=++sz;
fa[sz]=p;
}
p=to[p][c];
}
v[p]=1;
} il void build()
{
rg queue<int>Q;
Q.push(0);
while(!Q.empty())
{
rg int x=Q.front();
Q.pop();
v[x]|=v[nx[x]];
for(rg int i=0;i<CHARSET;++i)
{
if(to[x][i])
{
nx[to[x][i]]=x?to[nx[x]][i]:0;
Q.push(to[x][i]);
}
else
{
to[x][i]=x?to[nx[x]][i]:0;
}
}
}
for(rg int i=0;i<=sz;++i)
for(rg int j=0;j<CHARSET;++j)
{
a[to[i][j]].push_back(i);
}
memset(d,0x3f,sizeof d);
for(rg int i=0;i<=sz;++i)
if(v[i])
{
Q.push(i);
d[i]=0;
}
while(!Q.empty())
{
rg int x=Q.front();
Q.pop();
for(rg int i=0;i<a[x].size();++i)
{
rg int y=a[x][i];
if(d[y]>d[x]+1)
{
d[y]=d[x]+1;
Q.push(y);
}
}
}
}
}T; char s[MAXN],t[MAXN]; int f[MAXN],len; int main()
{
freopen("xining.in","r",stdin);
freopen("xining.out","w",stdout);
scanf("%s",s);
T.ins(s);
T.build();
scanf("%s",t);
printf("%d\n",d[f[len]]);
for(rg int i=0;t[i];++i)
{
if(t[i]=='-')
{
if(len>0)
--len;
}
else
{
rg int x=T.to[f[len]][t[i]-'a']; // edit 1
f[++len]=x;
}
// cerr<<"f="<<f[len]<<endl;
printf("%d\n",d[f[len]]);
}
// fclose(stdin);
// fclose(stdout);
return 0;
}

test20181020 B君的第二题的更多相关文章

  1. B 君的第二题 (hongkong)

    B 君的第二题 (hongkong) 题目大意: 一个长度为\(n(n\le2\times10^5)\)的数组,给定一个数\(k(k\le40)\).用\(a[i][j]\)表示该数组\(i\)次前缀 ...

  2. test20181016 B君的第二题

    题意 分析 考场暴力50分. 考虑bfs序,一个点的儿子节点的bfs序一定连续,所以对bfs序建线段树,努力打一下就行了. 时间复杂度\(O(n \log n + m \log n)\) #inclu ...

  3. test20181017 B君的第二题

    题意 分析 考场50分 旁边的L君告诉我,求的就是非升子序列的个数,于是写了个树状数组. 但是\(\mod{2333} > 0\)还需要组合数中没有2333的倍数,所以实际上只得了\(a_i \ ...

  4. test20181015 B君的第二题

    题意 分析 考场85分 用multiset暴力,由于教练的机子飞快,有写priority_queue水过了的人. #include<cstdlib> #include<cstdio& ...

  5. test20181020 B君的第一题

    题意 分析 二次剩余问题. x,y相当于二次方程 \[ x^2-bx+c=0 \mod{p} \] 的两根. 摸意义下的二次方程仍然考虑判别式\(\Delta=b^2-4c\). 它能开根的条件是\( ...

  6. test20181019 B君的第二题

    题意 分析 快速子集和变换以及快速超集和变换的裸题. 用\(f(s)\)表示集合s的方案数,初始化为输入中s出现的次数. 做一遍快速子集和变换,此时f(s)表示s及其子集在输入中出现的次数. 对所有f ...

  7. Java蓝桥杯02——第二题集锦:生日蜡烛、星期一、方格计数、猴子分香蕉

    第二题 生日蜡烛(结果填空) 某君从某年开始每年都举办一次生日party,并且每次都要吹熄与年龄相同根数的蜡烛. 现在算起来,他一共吹熄了236根蜡烛. 请问,他从多少岁开始过生日party的? 请填 ...

  8. 05:统计单词数【NOIP2011复赛普及组第二题】

    05:统计单词数 总时间限制:  1000ms 内存限制:  65536kB 描述 一般的文本编辑器都有查找单词的功能,该功能可以快速定位特定单词在文章中的位置,有的还能统计出特定单词在文章中出现的次 ...

  9. 常见面试第二题之什么是Context

    今天的面试题,也就是我们常见面试题系列的第二题,我们来讲一讲android中的context.我相信大家android开发者一定对于这个context非常熟悉,肯定都有使用过,肯定没有没使用过的.但是 ...

随机推荐

  1. linux体系结构与内核结构图解

    1.当被问到Linux体系结构(就是Linux系统是怎么构成的)时,我们可以参照下图这么回答:从大的方面讲,Linux体系结构可以分为两块: (1)用户空间:用户空间中又包含了,用户的应用程序,C库 ...

  2. 框架-spring源码分析(一)

    框架-spring源码分析(一) 参考: https://www.cnblogs.com/heavenyes/p/3933642.html http://www.cnblogs.com/BINGJJF ...

  3. Android下拉刷新控件--PullToRefresh的简单使用

    Android中很多时候都会用到上下拉刷新,这是一个很常用的功能,Android的v4包中也为我们提供了一种原生的下拉刷新控件--SwipeRefreshLayout,可以用它实现一个简洁的刷新效果, ...

  4. 关于浏览器的eventflow(capture and bubble up)

    因为,没有全面的学习javascript,及其事件原理: 全占的课程:4-5 浏览器 Bubble Up 事件模型中 不是很理解它所讲的.网上查找相关知识点.记录中在博客中: 理解了JS的加载 htt ...

  5. hdu3549网络流之最大流

    Ford-Fulkerson方法:dfs实现 dfs  140ms #include<map> #include<set> #include<cmath> #inc ...

  6. js将 HTML 页面生成 PDF 并下载

    最近碰到个需求,需要把当前页面生成 pdf,并下载.弄了几天,自己整理整理,记录下来,我觉得应该会有人需要 :) 先来科普两个插件: html2Canvas 简介 我们可以直接在浏览器端使用html2 ...

  7. C#删除图片问题

    public Image GetImage(string path) { FileStream fs = new FileStream(path, FileMode.Open, FileAccess. ...

  8. 手写DCGAN

    //加上了注释,对pytorch又加深了理解import torch as t from torch import nn from torch.autograd import Variable fro ...

  9. 将java打jar包成linux后台服务service

    将java打jar包成linux后台服务service 第一步:将java程序打成jar包 build.gradle配置文件中加spring-boot-gradle-plugin插件,具体配置如下(配 ...

  10. bzoj 1854 构图 并查集

    我们可以把一件装备看成一条边,两个属性看成两个点,那么这就相当于读入了一张图 当读入每一个x,y时,我们找到两个点的祖先节点,fx,fy,我们保证祖先节点在该连通块 中编号(装备属性)最大,用flag ...