正题

题目链接:https://www.luogu.com.cn/problem/P7518


题目大意

给出\(n\)个点的一棵树,每个点上有不大于\(m\)的数字。

然后给出一个长度为\(c\)的各个位数不同的序列,每次询问一条路径上找到一个最大的\(k\)使得该序列的存在\(1\sim k\)的子序列。

\(1\leq n,q\leq 2\times 10^5,1\leq c\leq m\leq 5\times 10^4,1\leq w_i\leq m\)


解题思路

传统的思想,路径分为向上和向下的两部分。然后因为序列没有重复元素,所以相当于对于每一种存在于序列的宝石都有唯一的下一种宝石。

先考虑向上的,发现我们必须从一开始,所以其实我们可以考虑离线记录一个\(last\)数组,其中\(last_i\)表示到根节点的路径中上一个\(i\)类型的是什么。

然后每个节点维护一棵线段树,对于节点\(x\)若是第\(i\)种宝石,那么第\(j\)个位置就储存它往上走到按顺序第\(i\sim j\)颗宝石的最大深度,这个可以每次从\(last_{i+1}\)处继承一棵树然后修改一个位置就好了。

然后询问的时候就直接从\(last_1\)处的树上二分出我们需要深度就可以确定我们往上走的路径能走到哪里了。

考虑向下的路径,我们把它拆成一条反向向上的路径,但是起点不是固定的,所以我们可以直接二分答案,然后在\(last_{mid}\)处向上走到\(LCA\)时,查看是否上和下的路径的序列有重复部分就好了。

时间复杂度\(O(n\log^2 n)\)


code

考场代码比较凌乱

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#include<cctype>
using namespace std;
const int N=2e5+10,T=18;
struct edge{
int to,next;
}a[N<<1];
int n,m,c,tot,w[N],p[N],ls[N],ans[N],lca[N];
int f[N][T+1],dep[N],las[N],rev[N],rt[N],up[N];
vector<int> vs[N],vt[N];
int read(){
int x=0,f=1;char c=getchar();
while(!isdigit(c)){if(c=='-')f=-f;c=getchar();}
while(isdigit(c)){x=(x<<1)+(x<<3)+c-'0';c=getchar();}
return x*f;
}
struct SegTree{
int cnt,w[N*20],ls[N*20],rs[N*20];
int Change(int x,int L,int R,int pos,int val){
int now=++cnt;w[now]=max(w[x],val);
if(L==R){ls[now]=rs[now]=0;return now;}
int mid=(L+R)>>1;
if(pos<=mid)ls[now]=Change(ls[x],L,mid,pos,val),rs[now]=rs[x];
else rs[now]=Change(rs[x],mid+1,R,pos,val),ls[now]=ls[x];
return now;
}
int Ask(int x,int L,int R,int k){
if(!x)return 0;
if(L==R)return L;int mid=(L+R)>>1;
if(w[rs[x]]<k)return Ask(ls[x],L,mid,k);
return Ask(rs[x],mid+1,R,k);
}
int Bsk(int x,int L,int R,int k){
if(!x)return c+1;
if(L==R)return L;int mid=(L+R)>>1;
if(w[ls[x]]<k)return Bsk(rs[x],mid+1,R,k);
return Bsk(ls[x],L,mid,k);
}
}Tr;
void addl(int x,int y){
a[++tot].to=y;
a[tot].next=ls[x];
ls[x]=tot;return;
}
void dfs(int x,int fa){
f[x][0]=fa;dep[x]=dep[fa]+1;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa)continue;
dfs(y,x);
}
return;
}
int LCA(int x,int y){
if(dep[x]>dep[y])swap(x,y);
for(int i=T;i>=0;i--)
if(dep[f[y][i]]>=dep[x])y=f[y][i];
if(x==y)return x;
for(int i=T;i>=0;i--)
if(f[x][i]!=f[y][i])x=f[x][i],y=f[y][i];
return f[x][0];
}
void calc(int x,int fa){
int P=p[w[x]];
if(P){
rev[x]=las[P];las[P]=x;
rt[x]=Tr.Change(rt[las[P+1]],0,c,P,dep[x]);
}
for(int i=0;i<vs[x].size();i++){
int id=vs[x][i];
up[id]=Tr.Ask(rt[las[1]],0,c,dep[lca[id]]);
}
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa)continue;
calc(y,x);
}
if(P)las[P]=rev[x];
return;
}
void solve(int x,int fa){
int P=p[w[x]];
if(P){
rev[x]=las[P];las[P]=x;
rt[x]=Tr.Change(rt[las[P-1]],1,c+1,P,dep[x]);
}
for(int i=0;i<vt[x].size();i++){
int id=vt[x][i],l=up[id]+1,r=c;
if(lca[id]==x){ans[id]=up[id];continue;}
while(l<=r){
int mid=(l+r)>>1;
int tmp=Tr.Bsk(rt[las[mid]],1,c+1,dep[lca[id]]+1);
if(tmp<=up[id]+1)l=mid+1;
else r=mid-1;
}
ans[id]=r;
}
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa)continue;
solve(y,x);
}
if(P)las[P]=rev[x];
return;
}
int main()
{
n=read();m=read();c=read();
for(int i=1;i<=c;i++){
int x=read();p[x]=i;
}
for(int i=1;i<=n;i++)w[i]=read();
for(int i=1;i<n;i++){
int x=read(),y=read();
addl(x,y);addl(y,x);
}
dfs(1,0);
for(int j=1;j<=T;j++)
for(int i=1;i<=n;i++)
f[i][j]=f[f[i][j-1]][j-1];
m=read();
for(int i=1;i<=m;i++){
int s=read(),t=read();
lca[i]=LCA(s,t);
vs[s].push_back(i);
vt[t].push_back(i);
}
calc(1,1);Tr.cnt=0;
solve(1,1);
for(int i=1;i<=m;i++)
printf("%d\n",ans[i]);
return 0;
}
/*
7 3 3
2 3 1
2 1 3 3 2 1 3
1 2
2 3
1 4
4 5
4 6
6 7
5
3 5
1 3
7 3
5 7
7 5
*/

P7518-[省选联考2021A/B卷]宝石【主席树,二分】的更多相关文章

  1. P7514-[省选联考2021A/B卷]卡牌游戏【贪心】

    正题 题目链接:https://www.luogu.com.cn/problem/P7514 题目大意 给出\(n\)个卡牌有\(a_i/b_i\),开始都是\(a_i\)朝上,将不超过\(m\)张卡 ...

  2. [省选联考 2020 A 卷] 组合数问题

    题意 [省选联考 2020 A 卷] 组合数问题 想法 自己在多项式和数论方面还是太差了,最近写这些题都没多少思路,看完题解才会 首先有这两个柿子 \(k*\dbinom{n}{k} = n*\dbi ...

  3. [省选联考 2021 A 卷] 矩阵游戏

    很巧妙的一个构造. 我是没有想到的. 自己的思维能力可能还是不足. 考虑先满足\(b\)对\(a\)的限制,把\(a\)的第一行和第一列设\(0\),推出这个\(a\). 接下来考虑对这个\(a\), ...

  4. luoguP6623 [省选联考 2020 A 卷] 树(trie树)

    luoguP6623 [省选联考 2020 A 卷] 树(trie树) Luogu 题外话: ...想不出来啥好说的了. 我认识的人基本都切这道题了. 就我只会10分暴力. 我是傻逼. 题解时间 先不 ...

  5. luoguP6620 [省选联考 2020 A 卷] 组合数问题(斯特林数)

    luoguP6620 [省选联考 2020 A 卷] 组合数问题(斯特林数) Luogu 题外话: LN切这题的人比切T1的多. 我都想到了组合意义乱搞也想到可能用斯特林数为啥还是没做出来... 我怕 ...

  6. luoguP6624 [省选联考 2020 A 卷] 作业题(莫比乌斯反演,矩阵树定理)

    luoguP6624 [省选联考 2020 A 卷] 作业题(莫比乌斯反演,矩阵树定理) Luogu 题外话: Day2一题没切. 我是傻逼. 题解时间 某种意义上说刻在DNA里的柿子,大概是很多人学 ...

  7. [四校联考P3] 区间颜色众数 (主席树)

    主席树 Description 给定一个长度为 N 颜色序列A,有M个询问:每次询问一个区间里是否有一种颜色的数量超过了区间的一半,并指出是哪种颜色. Input 输入文件第一行有两个整数:N和C 输 ...

  8. P7515-[省选联考 2021A卷]矩阵游戏【差分约束】

    正题 题目链接:https://www.luogu.com.cn/problem/P7515 题目大意 有一个\(n*m\)的矩形\(A\),然后给出一个\((n-1)*(m-1)\)的矩形\(B\) ...

  9. P7518 & 省选联考2021 宝石

    这是一篇极其简单连像我这样省三参加不了省选的蒟蒻都能看懂的题解 前置知识: 倍增LCA  二分 栈 题意 PS:这是一篇完全面向初学者的题解,会非常细,大佬请无视 题目传送门 没有思路的时候, 我们往 ...

随机推荐

  1. vue 报错http://eslint.org/docs/rules/xxxxx

    vue 对语法的要求过于严格所以编译的时候报下面这个错误 解决办法: bulid文件夹 -> webpack.base.conf.js 找到下面的代码: module: { rules: [ / ...

  2. 【spring 注解驱动开发】Spring AOP原理

    尚学堂spring 注解驱动开发学习笔记之 - AOP原理 AOP原理: 1.AOP原理-AOP功能实现 2.AOP原理-@EnableAspectJAutoProxy 3.AOP原理-Annotat ...

  3. MVVMLight学习笔记(一)---MVVMLight概述

    一.MVVM概述 MVVM是Model-View-ViewModel的简写,主要目的是为了解耦视图(View)和模型(Model). MVVM结构如下: 相对于之前把逻辑结构写在Code Behind ...

  4. linux(4)----------ssh config详解

    1.概述 ~~  config为了方便我们批量管理多个ssh ~~  config存放在~/.ssh/config                 .XX代表隐藏目录 ~~  config配置语法 2 ...

  5. jQuery中的属性过滤选择器(四、五):[attribute] 、[attribute=value]、[attribute!=value] 、[attribute^=value] 等

    <!DOCTYPE html> <html> <head> <title>属性过滤选择器</title> <meta http-equ ...

  6. 使用GZIP压缩网页内容(一)

    在JDK中提供了GZIP压缩,来压缩网页的内容,降低网络传输时候的字节数,到达浏览器端的时候,再解压,GZIP压缩之后传输耗费的流量大大降低,但是同时也不会降低用户体验. package day04; ...

  7. centos7 权限更改,所属用户及用户组更改

    2021-08-03 # 查看文件的权限 ll 第一个字符, "-" 表示是文件, "d" 表示是目录(directory) 后面 9 个字符每 3 个字符又作 ...

  8. Git使用教程六

    冲突的产生与解决 案例:模拟产生冲突. ①同事在下班之后修改了线上仓库的代码 注意:此时我本地仓库的内容与线上不一致的. 2.第二天上班的时候,我没有做git pull操作,而是直接修改了本地的对应文 ...

  9. MySQL主主互备不同步的解决方法

    MySQL主主互备不同步 首先在服务器上执行show slave satus;可以看到很多同步的参数: Master_Log_File: SLAVE中的I/O线程当前正在读取的主服务器二进制日志文件的 ...

  10. Linux下cat命令的使用

    1.普通用法-->查看文件内容 cat file_name 查看文件时的相关参数: 1.cat f1.txt,查看f1.txt文件的内容. 2.cat -n f1.txt,查看f1.txt文件的 ...