P5327-[ZJOI2019]语言【线段树合并,LCA】

正题

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


题目大意

给出nnn个点的一棵树,和mmm条路径,求有多少个点对至少存在一条路径经过它们。

1≤n,m≤1051\leq n,m\leq 10^51n,m105


解题思路

有一个很显然的性质,如果点zzzx→yx\rightarrow yxy的路径上,并且(x,z)(x,z)(x,z)不合法,那么(x,y)(x,y)(x,y)肯定不合法。

所以这样的对于一个节点xxx来说所有和它合法的点会形成一棵生成树,这棵生成树是哪来的也很好说,我们把所有经过xxx的路径s→ts\rightarrow tstsssttt存下来,构出一棵虚树,这棵虚树的大小就是对于这个点来说合法的点个数。

虚树的大小怎么求,这个方法很多,而我们尽量使用一种比较方便动态维护的方法,把所有点按照dfsdfsdfs序排序,假设节点xxx之后排的是yyy(定义第一个之前排的是最后一个),那么就是所有depy−deplca(x,y)dep_{y}-dep_{lca(x,y)}depydeplca(x,y)的和。

不难发现这个东西可以在序列上维护,同理的可以在线段树上维护,每个区间记录dfsdfsdfs最小的和最大的点就好了。

那么做法已经很显然了,一条路径经过的点我们可以用树上差分来做到也就是s↔lca(s,t)s\leftrightarrow lca(s,t)slca(s,t)t↔lca(s,t)t\leftrightarrow lca(s,t)tlca(s,t)的部分。

然后两个子树的信息合并的时候用线段树合并就好了。

写了个RMQRMQRMQ来快速求LCALCALCA

时间复杂度:O(nlog⁡n)O(n\log n)O(nlogn)


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
const int N=1e5+10,M=N*20*4;
struct node{int to,next;
}a[N<<1];
int n,m,tot,cnt,dfc,ls[N],dep[N],rt[N];
int dfn[N],rfn[N],rgn[N],lg[N<<1],f[N<<1][19];
vector<int> v[N],g[N];
long long ans;
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){dep[x]=dep[fa]+1;dfn[++dfc]=x;rfn[x]=dfc;rgn[x]=++cnt;f[cnt][0]=x;for(int i=ls[x];i;i=a[i].next){int y=a[i].to;if(y==fa)continue;dfs(y,x);f[++cnt][0]=x;}return;
}
int LCA(int x,int y){int l=rgn[x],r=rgn[y];if(l>r)swap(l,r);int z=lg[r-l+1];x=f[l][z];y=f[r-(1<<z)+1][z];return (dep[x]<dep[y])?x:y;
}
int calc(int x,int y){if(!x||!y)return 0;return dep[y]-dep[LCA(x,y)];
}
struct SegTree{int cnt,w[M],s[M],lp[M],rp[M],ls[M],rs[M];void Merge(int x,int ls,int rs){s[x]=s[ls]+s[rs]+calc(rp[ls],lp[rs]);lp[x]=lp[ls]?lp[ls]:lp[rs];rp[x]=rp[rs]?rp[rs]:rp[ls];return;}void Change(int &x,int L,int R,int pos,int val){if(!x)x=++cnt;if(L==R){w[x]+=val;if(w[x])lp[x]=rp[x]=dfn[pos];else lp[x]=rp[x]=0;return;}int mid=(L+R)>>1;if(pos<=mid)Change(ls[x],L,mid,pos,val);else Change(rs[x],mid+1,R,pos,val);Merge(x,ls[x],rs[x]);return;}int Merge(int x,int y,int L,int R){if(!x||!y)return x+y;if(L==R){w[x]=w[x]+w[y];if(w[x])lp[x]=rp[x]=dfn[L];else lp[x]=rp[x]=0;return x;}int mid=(L+R)>>1;ls[x]=Merge(ls[x],ls[y],L,mid);rs[x]=Merge(rs[x],rs[y],mid+1,R);Merge(x,ls[x],rs[x]);return x;}
}T;
void solve(int x,int fa){for(int i=ls[x];i;i=a[i].next){int y=a[i].to;if(y==fa)continue;solve(y,x);rt[x]=T.Merge(rt[x],rt[y],1,n);}for(int i=0;i<v[x].size();i++)T.Change(rt[x],1,n,v[x][i],1);ans+=T.s[rt[x]]+dep[T.lp[rt[x]]]-dep[LCA(T.lp[rt[x]],T.rp[rt[x]])]+1;for(int i=0;i<g[x].size();i++)T.Change(rt[x],1,n,g[x][i],-2);return;
}
int main()
{scanf("%d%d",&n,&m);for(int i=1,x,y;i<n;i++){scanf("%d%d",&x,&y);addl(x,y);addl(y,x);}dfs(1,0);for(int j=1;(1<<j)<=cnt;j++)for(int i=1;i+(1<<j)-1<=cnt;i++){int x=f[i][j-1],y=f[i+(1<<j-1)][j-1];f[i][j]=(dep[x]<dep[y])?x:y;}for(int i=2;i<=cnt;i++)lg[i]=lg[i>>1]+1;for(int i=1,x,y;i<=m;i++){scanf("%d%d",&x,&y);v[x].push_back(rfn[x]);v[x].push_back(rfn[y]);v[y].push_back(rfn[x]);v[y].push_back(rfn[y]);int lca=LCA(x,y);g[lca].push_back(rfn[x]);g[lca].push_back(rfn[y]);}solve(1,0);printf("%lld\n",(ans-n)/2ll);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/317772.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

D. Bananas in a Microwave

D. Bananas in a Microwave 题意&#xff1a; u1s1&#xff0c;题意真难懂 有n个时间&#xff0c;每个时间给你两个操作&#xff0c;第一个是kkx&#xff0c;第二个是kk∗x&#xff0c;且可以执行[0,y]次&#xff0c;&#xff08;在第i个时间点&#xff0c;必须应用第i个操作…

.NET西安社区 [拥抱开源,又见 .NET] 第二次活动简报

「拥抱开源, 又见 .NET」随着 .NET Core的发布和开源&#xff0c;.NET又重新回到人们的视野。 .NET Core的下个3.0即将release&#xff0c;加入非常多的新功能&#xff0c;越来越拥抱变化&#xff0c;DevOps和Microservice的最佳实践已经在 .NET Core落地&#xff0c;比如 Ocel…

YBTOJ洛谷P2839:最大中位数(主席树、二分答案)

遇事不决&#xff0c;二分试试 解析 很好的一道题 真是把主席树玩明白了 一个关于中位数的常用trick&#xff1a; 二分答案mid&#xff0c;把>mid的看成1&#xff0c;<mid的看成-1&#xff0c;然后看最大子段和是否>0 然而如果对离散化后的每一个值建一棵小白逛公园那…

计数学习小记

前言 闲的无聊懒得做题不如来水点博客。 虽然一直作为一个感性做题的选手&#xff0c;但是理性层面上确实是分析题目初步做法的一个十分重要的方法。 额不会涉及具体的知识点&#xff0c;只是总结点自己做题的时候遇到的比较巧妙的方法。 混沌排版请见谅 还有我也很菜有错…

[dsu on tree]树上启发式合并总结(算法思想及模板附例题练习)

文章目录前言树上启发式合并引入算法思想时间复杂度模板练习例题&#xff1a;CF600E Lomsat gelralsolutioncodeCF208E Blood CousinssolutioncodeCF570D Tree RequestssolutioncodeCF1009F Dominant Indicessolutioncode前言 最近不是在⛏李超树嘛&#xff0c;然后就去玩了下…

F Christmas Game

F Christmas Game 题意&#xff1a; 给一棵n个节点树&#xff0c;每个点上都有权值&#xff0c;两个人轮流操作&#xff0c;每次可以将一个点的权值给他的父亲节点&#xff0c;&#xff08;父亲节点与当前点的深度差必须为k&#xff09;&#xff0c;当有一方不能操作时即为输…

领域驱动设计,让程序员心中有码(七)

领域驱动设计- 让程序员心中有码&#xff08;七&#xff09;-设计原则和设计模式&#xff0c;互联网开发者们共同的追求前言多年来&#xff0c;笔者一直从事传统软件企业的软件开发和项目管理工作。笔者发现在众多的传统软件企业中&#xff0c;评判优秀开发者的标准往往是技能的…

codeforces:CF750 复盘

总结 solve 4 rnk:833 我变成小学生(pupil)啦&#xff01; 这次看完题解感觉D和F其实都挺可做的 还是思维的问题 A 签到题 判断加起来是奇数还是偶数即可 但我做的分类讨论就很恶心qwq 因为我没看到a、b、c都至少是1&#xff01; 所以我还在辛苦的分类讨论qwq 只能说我分类讨…

UOJ#351-新年的叶子【树的直径,数学期望】

正题 题目链接:https://uoj.ac/problem/351 题目大意 给出nnn个点的一棵树&#xff0c;开始所有点都是白色&#xff0c;每次随机点黑一个叶子&#xff08;可以重复点&#xff09;&#xff0c;求期望多少次能使得白色点构成的图直径发生变化。 答案对998244353998244353998244…

cf1504. Travelling Salesman Problem

cf1504. Travelling Salesman Problem 题意&#xff1a; n个城市&#xff0c;编号1~n&#xff0c;每个城市有美丽值a[i]&#xff0c;现在要从城市1出发&#xff0c;其他所有城市走一遍&#xff0c;最后回到城市1&#xff0c;城市i到j的花费为max(ci,aj-ai)&#xff0c;ci为第…

[NOIP-S 2020]游记(附考前注意事项)

呜呼起飞T1&#xff1a;排水系统T2&#xff1a;字符串匹配T3&#xff1a;移球游戏T4&#xff1a;微信步数总述考前注意事项T1&#xff1a;排水系统 嗯—— 怎么说呢&#xff1f;&#xff1f; 比赛开始后迅速通读三遍题 顶着第一题肯定是打卡题的心态 哪怕是恶心模拟也得上&…

模板:二维线段树(线段树套线段树)

文章目录问题解析单点修改询问完整代码标记永久化代码所谓二维线段树&#xff0c;就是有两个维度的线段树 (逃) 问题 给出一个矩形 要求支持以下操作&#xff1a; 1.询问一个子矩形的最值 2.修改某一个单点的值 解析 使用线段树套线段树&#xff0c;来解决二维动态问题 注意…

程序猿修仙之路--数据结构之你是否真的懂数组?

数据结构但凡IT江湖侠士&#xff0c;算法与数据结构为必修之课。早有前辈已经明确指出&#xff1a;程序算法数据结构 。要想在之后的江湖历练中通关&#xff0c;数据结构必不可少。数据结构与算法相辅相成&#xff0c;亦是阴阳互补之法。开篇说道数组&#xff0c;几乎每个IT江…

P5643-[PKUWC2018]随机游走【min-max容斥,dp】

正题 题目链接:https://www.luogu.com.cn/problem/P5643 题目大意 给出nnn个点的一棵树&#xff0c;一个人从点xxx开始随机游走&#xff0c;然后QQQ次询问给出一个点集SSS&#xff0c;求期望多少步这个人会经过这个点集中的所有点。 1≤n≤18,1≤Q≤50001\leq n\leq 18,1\leq…

Rolling The Polygon Gym - 102222B

Rolling The Polygon Gym - 102222B 题意&#xff1a; 给你一个多边形&#xff0c;给你内部一个点Q&#xff0c;多边形在平面上滚动一周&#xff08;当有一个边第二次触地滚动停止&#xff09;&#xff0c;问Q的轨迹长度 题解&#xff1a; 计算几何题目 自己一直不是很擅长…

[杂题训练]CF1228E Another Filling the Grid(容斥),CF936C Lock Puzzle(构造)

文章目录T1&#xff1a;CF1228E Another Filling the GridsolutioncodeT2&#xff1a;CF936C Lock PuzzlesolutioncodeT1&#xff1a;CF1228E Another Filling the Grid 点我 solution 反过来思考&#xff0c;用所有方案数➖不合法方案数 很容易想到的是——容斥&#xff01…

Asp.Net Core 轻松学-经常使用异步的你,可能需要看看这个文章

前言事情的起因是由于一段简单的数据库连接代码引起&#xff0c;这段代码从语法上看&#xff0c;是没有任何问题&#xff1b;但是就是莫名其妙的报错了&#xff0c;这段代码极其简单&#xff0c;就是打开数据库连接&#xff0c;读取一条记录&#xff0c;然后立即更新到数据库中…

CF1146F: Leaf Partition(树形dp)

解析 阴间dp题qwq 不难设计dp&#xff1a; dpx,0:x节点没有被包含、子树内的方案数dp_{x,0}:x节点没有被包含、子树内的方案数dpx,0​:x节点没有被包含、子树内的方案数 dpx,1:x节点被包含、子树内的方案数dp_{x,1}:x节点被包含、子树内的方案数dpx,1​:x节点被包含、子树内的…

Take Your Seat Gym - 102222D

Take Your Seat Gym - 102222D 题意&#xff1a; 第一次是n个人坐飞机&#xff0c;按照1到n的顺序登机&#xff0c;第一个人登机牌丢了&#xff0c;他随机做一个座位&#xff0c;2到n个人上来按照自己的登机牌坐座位&#xff0c;如果他的座位被坐了&#xff0c;就在随机找一个…

CF446D-DZY Loves Games【高斯消元,矩阵乘法】

正题 题目链接:https://www.luogu.com.cn/problem/CF446D 题目大意 给出nnn个点mmm条边的一张无向图&#xff0c;一些点有陷阱&#xff0c;走到时会损失一条生命&#xff0c;总共有kkk条生命&#xff0c;求从111出发随机游走到nnn没有死亡且到终点时仅剩一条命的概率。 1≤n≤…