Link

有两种做法:

第一种是\(O(nlog\ n)\)的。

我们预处理两个数组:

\(pre_i\)表示\(p\)中\(i\)前面的那个数是\(pre_i\)。

\(lst_i\)表示\(a\)中\(a_i\)前一个\(pre_{a_i}\)的位置。(代码中是\(f_0\))

那么每个数往前跳几次\(lst\),也就会对应排列\(p\)中的一段连续子串。

然后处理往前跳\(lst\)的倍增数组。

我们知道,\(p\)是一个循环排列,所以对于每个数,以这个数结束的最短的是\(p\)的循环移位的\(a\)的子序列,就是从这个数往前跳\(n-1\)次\(lst\)得到的子序列。

所以我们处理出每个点往前跳\(n-1\)次\(lst\)得到的位置\(edp\)。

对于一次询问\(l,r\),如果\(l,r\)中存在子序列为\(p\)的循环移位,那么就会满足\(\exist i\in[l,r],edp_i\in[l,r]\)。

所以我们用\(st\)表维护区间最小值即可。

#include<bits/stdc++.h>
using namespace std;
const int N=200007;
int p[N],a[N],f[21][N],pre[N],lst[N],Log[N],st[21][N];
int max(int a,int b){return a>b? a:b;}
int read(){int x;scanf("%d",&x);return x;}
int get(int l,int r){int k=Log[r-l+1];return max(st[k][l],st[k][r-(1<<k)+1]);}
int main()
{
int n=read(),m=read(),q=read(),i,j,l,r;
for(i=1;i<=n;++i) p[i]=read();
p[0]=p[n];
for(i=1;i<=n;++i) pre[p[i]]=p[i-1];
for(i=1;i<=m;++i) a[i]=read();
for(i=1;i<=m;++i) f[0][i]=lst[pre[a[i]]],lst[a[i]]=i;
for(i=2;i<=n||i<=m;++i) Log[i]=Log[i>>1]+1;
for(i=1;i<=Log[n];++i) for(j=1;j<=m;++j) f[i][j]=f[i-1][f[i-1][j]];
for(i=1;i<=m;++i) for(st[j=0][i]=i;j<=Log[n-1];++j) if((n-1)&(1<<j)) st[0][i]=f[j][st[0][i]];
for(i=1;i<=Log[m];++i) for(j=1;j+(1<<i)-1<=m;++j) st[i][j]=max(st[i-1][j],st[i-1][j+(1<<i-1)]);
while(q--) l=read(),r=read(),putchar(get(l,r)>=l? '1':'0');
}

第二种是\(O(n)\)的。

我们处理两个数组:

\(ppre_i\)表示\(p\)中\(i\)前面的那个数是\(ppre_i\)。

\(apre_i\)表示\(a\)中前一个\(a_i\)的位置。

然后从前往后扫\(a\)数组,把\(a_i\)前面所有未连边的\(ppre_{a_i}\)的位置向\(i\)连边,并且打上标记,下次不再连边。

那么我们知道每个点最多只有一个入边。没有入边的,我们新建一个\(0\)到这个点的边。

总的,我们会建出一棵树。

\(a\)数组中\(a_i\)向前跳一次\(lst\)(上面那个做法的跳)也就是跳一次父亲。

所以上面做法的跳\(n-1\)次就变成了\(n-1\)级祖先。

考虑dfs的过程,dfs到某个点经过的点(也就是dfs的栈)存的都是根到当前节点的路径。

所以每个节点dfs到时(进栈)把当前深度的节点(\(b\)数组)记为自己,因此\(n-1\)级祖先就是\(b[dep[u]-(n-1)]\)。

#include<bits/stdc++.h>
#define pb push_back
using namespace std;
const int N=200007;
int n,m,q,p[N],a[N],apre[N],lst[N],ppre[N],fa[N],dep[N],w[N],b[N];
vector<int>G[N];
int max(int a,int b){return a>b? a:b;}
int min(int a,int b){return a<b? a:b;}
int read(){int x;scanf("%d",&x);return x;}
void dfs(int u)
{
int i,v;
b[dep[u]]=u,w[u]=dep[u]>=n? b[dep[u]-(n-1)]:m+1;
for(i=G[u].size()-1;~i;--i) v=G[u][i],dep[v]=dep[u]+1,dfs(v);
}
int main()
{
n=read(),m=read(),q=read();int i,j;
for(i=1;i<=n;++i) p[i]=read();
p[0]=p[n];
for(i=1;i<=n;++i) ppre[p[i]]=p[i-1];
for(i=1;i<=m;++i)
{
a[i]=read(),apre[i]=lst[a[i]],lst[a[i]]=i;
for(j=lst[ppre[a[i]]];j&&!fa[j];j=apre[j]) fa[j]=i;
}
for(i=1;i<=m;++i) fa[i]? G[fa[i]].pb(i):G[0].pb(i);
dfs(0);
for(i=m-1;i;--i) w[i]=min(w[i+1],w[i]);
while(q--) i=read(),j=read(),putchar(48+(w[i]<=j));
}

CF1142B Lynyrd Skynyrd的更多相关文章

  1. 【题解】CF1142B Lynyrd Skynyrd(倍增)

    [题解]CF1142B Lynyrd Skynyrd(倍增) 调了一个小时原来是读入读反了.... 求子段是否存在一个排列的子序列的套路是把给定排列看做置换,然后让给定的序列乘上这个置换,问题就转化为 ...

  2. 【CF1142B】Lynyrd Skynyrd

    [CF1142B]Lynyrd Skynyrd 题面 洛谷 题解 假设区间\([l,r]\)内有一个循环位移,那么这个循环位移一定有一个最后的点,而这个点在循环位移中再往前移\(n-1\)个位置也一定 ...

  3. CF1142E/1143B Lynyrd Skynyrd

    CF1142E/1143B Lynyrd Skynyrd 开始读错题了,以为是连续的一段,敲完后才发现是 \(subsequence\) ... 考虑对于 \(a\) 中的每个 \(a_i\) 找到它 ...

  4. 「CF1142B」Lynyrd Skynyrd

    传送门 Luogu 解题思路 发现一个性质: 对于排列的任何一个循环位移,排列中的同一个数的前驱肯定是不变的. 而且,如果一个排列的循环位移是某一个区间的子序列,那么这个循环位移的结尾的 \(n-1\ ...

  5. 『题解』Codeforces1142B Lynyrd Skynyrd

    更好的阅读体验 Portal Portal1: Codeforces Portal2: Luogu Description Recently Lynyrd and Skynyrd went to a ...

  6. B. Lynyrd Skynyrd

    传送门: 题意:给出 n,m,q 然后给出模板串,从1-n数字只出现一次,然后给出长度为m的要询问的串. q组询问:每组询问输出 ‘1’或者‘0’ 每组询问 一对x,y    问在x到y中有没有模板串 ...

  7. Codeforces 1142B Lynyrd Skynyrd

    ---恢复内容开始--- 题意:给你一个排列p和数组a,有t组询问,每次询问一个区间的子序列中是否有p的一个循环排列. 思路:以p = [3, 1, 2]举例, 我们扫描数组b,假设当前数字是1,那么 ...

  8. Codeforces Round #549 (Div. 1)

    今天试图用typora写题解 真开心 参考 你会发现有很多都是参考的..zblzbl Codeforces Round #549 (Div. 1) 最近脑子不行啦 需要cf来缓解一下 A. The B ...

  9. Codeforces Round #549 (Div. 2) 训练实录 (5/6)

    The Doors +0 找出输入的01数列里,0或者1先出完的的下标. Nirvana +3 输入n,求1到n的数字,哪个数逐位相乘的积最大,输出最大积. 思路是按位比较,从低到高,依次把小位换成全 ...

随机推荐

  1. 顺序查找(Sequential Search)

    1.定义 顺序查找又叫线性查找,是最基本的查找技术. 2.基本思想 从表的一端开始(第一个或最后一个记录),顺序扫描线性表,依次将扫描到的结点关键宇和给定值K相比较.若当前扫描到的结点关键字与K相等, ...

  2. 5.JavaBean

    JavaBean JSP开发初期HTML,css,java代码混杂在一起,给程序的调试和维护带来很大困难.将与HTML分离并将对象和逻辑java代码封装成类就是一个JavaBean组件. 1.Java ...

  3. postfix -- 发件调试

    按照教程(https://www.cnblogs.com/huandada/p/10554603.html)搭建好postfix之后,由于收件的邮件运营商的限制,部分邮件不能正常发送,需要更多其他配置 ...

  4. 微信小程序_(视图)简单的scroll-view容器

    scroll-view容器效果 官方文档:传送门 scroll-view 可滚动视图区域 scroll-x Boolean false 允许横向滚动 scroll-y Boolean false 允许 ...

  5. @ControllerAdvice 全局异常处理

    使用@ControllerAdvice 定义 全局异常处理 package com.app; import java.io.IOException; import java.io.PrintWrite ...

  6. ubuntu环境配置终极解答

    1. ubuntu中常用的5个配置文件 1)/etc/profile 2)/etc/environment 环境变量在这个文件中定义,可以用vim /etc/environment查看该文件内容 3) ...

  7. i 是一个修饰符 (搜索不区分大小写)

    什么是正则表达式? 正则表达式是由一个字符序列形成的搜索模式. 当你在文本中搜索数据时,你可以用搜索模式来描述你要查询的内容. 正则表达式可以是一个简单的字符,或一个更复杂的模式. 正则表达式可用于所 ...

  8. sql server关键字大全

    保留关键字 Microsoft® SQL Server™ 2000 使用保留关键字定义.操作和访问数据库.保留关键字是 SQL Server 使用的 Transact-SQL 语言语法的一部分,用于分 ...

  9. 【Spark机器学习速成宝典】基础篇04数据类型(Python版)

    目录 Vector LabeledPoint Matrix 使用C4.5算法生成决策树 使用CART算法生成决策树 预剪枝和后剪枝 应用:遇到连续与缺失值怎么办? 多变量决策树 Python代码(sk ...

  10. 一起学vue指令之v-text

    一起学vue指令之v-text 一起学 vue指令 v-text  指令可看作标签属性 v-text的功能和v-html很相似,都是在容器标签内控制字符串内容的输出,v-text输出纯文本,而v-ht ...