博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
[CF191](Fools and Roads)
阅读量:5054 次
发布时间:2019-06-12

本文共 1070 字,大约阅读时间需要 3 分钟。

题意 : 给你一棵树,然后给你m对点,将每对点之间的最短路径上每条边权值+1,求操作完成后每条边的权值

solution:树上差分(其实如果你是数据结构大师的话也可以用树链剖分做)

树上差分的板子是这样的:

设差分数组p,对于路径s->t,p[s]++,p[t]++,p[lca(s,t)]--,p[fa[lca[(s,t)]]]--;

然后一个点的子树内差分数组值之和即为该点被覆盖的次数

然而这题要求我们处理边

那么我们有两种方法

一种是对于一条边,新建一个点代表这条边,由该点向边的两个端点连边

暴力但很无脑

另一种是用一条边的两个端点中深度较大的端点代表这条边

但此时原来的差分操作会出锅,要改为p[s]++,p[t]++,p[lca(s,t)]-=2(自己理解)

贴代码(第一种方法,欧拉序ST表求LCA)

#include
#include
#include
#include
#include
#define N 400050using namespace std;vector
G[N];int n,m;int dfn[N],pos[N],lg[N],dep[N];int id[N],ans[N];int st[35][N],cnt=0,plu[N],fa[N];void aux(int x,int ff) { dfn[++cnt]=x,pos[x]=cnt,fa[x]=ff; for(int i=0; i
>1]+1; for(int i=1; i<=cnt; i++)st[0][i]=dfn[i]; for(int i=1; i<=lg[cnt]; i++) for(int r=1; r+(1<
<=cnt; r++) st[i][r]=mn(st[i-1][r],st[i-1][r+(1<<(i-1))]); }int lca(int a,int b) { int x=pos[a],y=pos[b]; if(x>y)swap(x,y); int p=lg[y-x+1]; return mn(st[p][x],st[p][y-(1<

转载于:https://www.cnblogs.com/stepsys/p/10802586.html

你可能感兴趣的文章
Activity和Fragment生命周期对比
查看>>
查找 EXC_BAD_ACCESS 问题根源的方法
查看>>
日常报错
查看>>
list-style-type -- 定义列表样式
查看>>
Ubuntu 编译出现 ISO C++ 2011 不支持的解决办法
查看>>
Linux 常用命令——cat, tac, nl, more, less, head, tail, od
查看>>
VueJS ElementUI el-table 的 formatter 和 scope template 不能同时存在
查看>>
Halcon一日一练:图像拼接技术
查看>>
iOS设计模式 - 中介者
查看>>
centos jdk 下载
查看>>
HDU 1028 Ignatius and the Princess III(母函数)
查看>>
(转)面向对象最核心的机制——动态绑定(多态)
查看>>
token简单的使用流程。
查看>>
django创建项目流程
查看>>
Vue 框架-01- 入门篇 图文教程
查看>>
多变量微积分笔记24——空间线积分
查看>>
poi操作oracle数据库导出excel文件
查看>>
(转)Intent的基本使用方法总结
查看>>
Windows Phone开发(24):启动器与选择器之发送短信
查看>>
JS截取字符串常用方法
查看>>