UVA1265 Tour Belt Kruskal重构树、倍增、树上差分
题意:定义$Tour \, Belt$为某张图上的一个满足以下条件的点集:①点集中至少有$2$个点②任意两点互相连通③图上两个端点都在这个点集中的边的权值的最小值严格大于图上只有一个端点在这个点集中的边的权值的最大值。现在给你一张$N$个点,$M$条边的图,请给出这张图上所有$Tour\,Belt$中包含的点数的和。$N \leq 5000 , M \leq \frac{N(N - 1)}{2}$
虽然这道题没有必要用$Kruskal$重构树来写,但是考%你赛的时候写$Kruskal$重构树的时候写挂了(LCA写错了),所以补上这个坑
因为我们需要某个连通块中所有边都大于其连向外面的边,所以考虑使用最大生成树的加边方式,每一次形成一个新的连通块的时候统计一次答案。可是我们不知道这个连通块中是否还有边没有被加进去,无法知道那些没有被加入的边是否会导致这个连通块的贡献变为$0$,所以实现有些麻烦(实际上根据这一点可以写出$O(N^2)$的并查集写法,然而咕咕咕咕咕)
我们自然地想到使用$Kruskal$重构树解决这个问题。在$Kruskal$重构树中,每个非叶子节点对应一个$Kruskal$过程中形成的连通块,而它的贡献就是这个子树中的叶子节点的个数。我们在非叶子节点中额外记录一个合并时的边权。在$Kruskal$重构树建好后,一遍$dfs$建好树上倍增,然后我们就可以考虑非树边对现有的连通块的影响了。
首先我们可以知道,在$Kruskal$重构树上,某条边的影响范围从两个端点的$LCA$开始,而它会使根到$LCA$路径上父亲对应边权大于等于当前边边权的点失去贡献,因为其父亲合并时的边权就是当前连通块中只连一个端点的边中最大的边权。可以知道这在$Kruskal$重构树上是一条链。我们就可以使用树上倍增找到满足这个条件的深度最低的点,通过树上差分对这一条链打上标记,最后遍历整棵树统计答案。
时间复杂度为$O(MlogN)$,理论上过不了实际上跑得飞快???
我的第一棵$Kruskal$重构树qaq
#include<bits/stdc++.h>
#define MAXM 1000010
#define MAXN 5010
using namespace std;
inline int read() {
;
;
char c = getchar();
while(!isdigit(c)) {
if(c == '-')
f = ;
c = getchar();
}
while(isdigit(c)) {
a = (a << ) + (a << ) + (c ^ ');
c = getchar();
}
return f ? -a : a;
}
struct Edge {
int start , end , w;
} Ed[MAXM];
][] , ch[MAXN << ][] , size[MAXN << ] , fa[MAXN << ] , dep[MAXN << ] , tag[MAXN << ] , now[MAXN << ];
int N , M , cntNode , ans;
bool vis[MAXM];
bool cmp(Edge a , Edge b) {
return a.w > b.w;
}
int find(int a) {
return fa[a] == a ? a : (fa[a] = find(fa[a]));
}
void dfs(int now , int fa) {
dep[now] = dep[fa] + ;
to[now][] = fa;
; i <= ; i++)
to[now][i] = to[to[now][i - ]][i - ];
]) {
dfs(ch[now][] , now);
dfs(ch[now][] , now);
}
}
inline int jumpToLCA(int p , int q) {
if(dep[p] < dep[q])
swap(p , q);
; i >= ; i--)
<< i) >= dep[q])
p = to[p][i];
if(p == q)
return p;
; i >= ; i--)
if(to[p][i] != to[q][i]) {
p = to[p][i];
q = to[q][i];
}
];
}
int jump(int n , int w) {
; i >= ; i--)
if(now[to[n][i]] >= w)
n = to[n][i];
return n;
}
int Dfs(int now) {
]) {
]) + Dfs(ch[now][]);
)
ans += size[now];
tag[now] = ;
return t;
}
;
}
int main() {
for(int T = read() ; T ; T--) {
dep[] = ans = ;
memset(vis , , sizeof(vis));
memset(ch , , sizeof(ch));
memset(now , , sizeof(now));
cntNode = N = read();
M = read();
; i <= N ; i++) {
fa[i] = i;
size[i] = ;
}
; i <= M ; i++) {
Ed[i].start = read();
Ed[i].end = read();
Ed[i].w = read();
}
sort(Ed + , Ed + M + , cmp);
; i <= M ; i++) {
int p = find(Ed[i].start) , q = find(Ed[i].end);
if(p != q) {
fa[p] = fa[q] = fa[cntNode] = ++cntNode;
ch[cntNode][] = p;
ch[cntNode][] = q;
size[cntNode] = size[p] + size[q];
now[cntNode] = Ed[i].w;
if(now[p] == Ed[i].w){
tag[p]--;
tag[cntNode]++;
}
if(now[q] == Ed[i].w){
tag[q]--;
tag[cntNode]++;
}
vis[i] = ;
}
}
dfs(cntNode , cntNode);
; i <= M ; i++)
if(!vis[i]) {
int t = jumpToLCA(Ed[i].end , Ed[i].start);
int p = jump(t , Ed[i].w);
tag[t]--;
tag[p]++;
}
Dfs(cntNode);
cout << ans << endl;
}
;
}
题外话:
论把
inline int jumpToLCA(int p , int q) {
if(dep[p] < dep[q])
swap(p , q);
; i >= ; i--)
<< i) >= dep[q])
p = to[p][i];
if(p == q)
return p;
; i >= ; i--)
if(to[p][i] != to[q][i]) {
p = to[p][i];
q = to[q][i];
}
];
}
写成
inline int jumpToLCA(int p , int q) {
if(dep[p] < dep[q])
swap(p , q);
; i >= ; i--)
<< i) >= q)
p = to[p][i];
if(p == q)
return p;
; i >= ; i--)
if(to[p][i] != to[q][i]) {
p = to[p][i];
q = to[q][i];
}
];
}
考%你赛还查不出来qwq
UVA1265 Tour Belt Kruskal重构树、倍增、树上差分的更多相关文章
- BZOJ5415[Noi2018]归程——kruskal重构树+倍增+堆优化dijkstra
题目描述 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 n 个节点.m 条边的无向连通图(节点的编号从 1 至 n).我们依次用 l,a 描述一条边的长度.海 ...
- NOI2018Day1T1 归程 并查集 kruskal kruskal重构树 倍增表 Dijkstra
原文链接https://www.cnblogs.com/zhouzhendong/p/NOI2018Day1T1.html 题目传送门 - 洛谷P4768 题意 给定一个无向连通图,有 $n$ 个点 ...
- BZOJ3732Network——kruskal重构树+倍增+LCA/最小生成树+倍增
题目描述 给你N个点的无向图 (1 <= N <= 15,000),记为:1…N. 图中有M条边 (1 <= M <= 30,000) ,第j条边的长度为: d_j ( 1 & ...
- LOJ #2718. 「NOI2018」归程(Dijkstra + Kruskal重构树 + 倍增)
题意 给你一个无向图,其中每条边有两个值 \(l, a\) 代表一条边的长度和海拔. 其中有 \(q\) 次询问(强制在线),每次询问给你两个参数 \(v, p\) ,表示在 \(v\) 出发,能开车 ...
- 【BZOJ 3732】 Network Kruskal重构树+倍增LCA
Kruskal重构树裸题, Sunshine互测的A题就是Kruskal重构树,我通过互测了解到了这个神奇的东西... 理解起来应该没什么难度吧,但是我的Peaks连WA,,, 省选估计要滚粗了TwT ...
- LOJ.2718.[NOI2018]归程(Kruskal重构树 倍增)
LOJ2718 BZOJ5415 洛谷P4768 Rank3+Rank1无压力 BZOJ最初还不是一道权限题... Update 2019.1.5 UOJ上被hack了....好像是纯一条链的数据过不 ...
- BZOJ 4242: 水壶(Kruskal重构树 + Bfs)
题意 一块 \(h ∗ w\) 的区域,存在障碍.空地.\(n\) 个建筑,从一个建筑到另一个建筑的花费为:路径上最长的连续空地的长度. \(q\) 次询问:从建筑 \(s_i\) 到 \(t_i\) ...
- [luogu4768] [NOI2018] 归程 (Dijkstra+Kruskal重构树)
[luogu4768] [NOI2018] 归程 (Dijkstra+Kruskal重构树) 题面 题面较长,这里就不贴了 分析 看到不能经过有积水的边,即不能经过边权小于一定值的边,我们想到了kru ...
- Luogu P4768 [NOI2018]归程(Dijkstra+Kruskal重构树)
P4768 [NOI2018]归程 题面 题目描述 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 \(n\) 个节点. \(m\) 条边的无向连通图(节点的编 ...
随机推荐
- AppBoxPro(权限管理框架--FineUIPro基础版+工厂模式+ADO.NET+存储过程)
FineUIPro基础版火爆来袭,特献上ADO.NET纯SQL方式AppBoxPro,希望大家能够喜欢! 下载源码请到[知识星球] https://t.zsxq.com/3rrNFyv
- 快捷键整理(来源:http://www.cnblogs.com/xing901022/p/4741630.htm)
Eclipse 跳转到指定行:ctrl+l 1几个最重要的快捷键 代码助手:Ctrl+Space(简体中文操作系统是Alt+/)快速修正:Ctrl+1单词补全:Alt+/打开外部Java文档:Shif ...
- 安卓开发_数据存储技术_SharedPreferences类
SharedPreferences类 供开发人员保存和获取基本数据类型的键值对. 该类主要用于基本类型,例如:booleans,ints,longs,strings.在应用程序结束后,数据仍旧会保存. ...
- spring 引用Bean的属性值
引用Bean的属性值 从Spring3.0开始,可以通过#{beanName.beanProp}的方式方便地引用另一个bean的属性值1.不需要使用PropertyPlaceholderConfigu ...
- 使用openssl在windows 10下本地xampp配置https开发环境
安装win64OpenSSL-1_1_0j后重新启动:以管理员权限启动powershell; 执行以下命令 set OPENSSL_CONF=c:\xampp\apache\conf\openssl. ...
- linux下zip文件解压乱码的问题
因为编码问题,zip文件中的中文文件在linux下解压会出现乱码 如果你使用archlinux那么使用AUR安装unzip-natspec就可以解决这个问题 https://aur.archlinux ...
- MSSQL清理所有用户数据库日志(SQLSERVER2008)
USE [master]; SET NOCOUNT ON; )=''; )=''; DECLARE @clearSql VARCHAR(MAX)=''; ; ,),TMP_WHILE_FLAG, T. ...
- windows下设置JupyterNotebook默认目录
目录 windows下设置JupyterNotebook默认目录 生成配置文件 设置默认工作目录 设置快捷方式中的目标与起始位置 直接修改anaconda中的相关配置文件 windows下设置Jupy ...
- 【PAT】B1042 字符统计(20 分)
/* 15分的题很简单,但是自己写的时候在输入数据时没有考虑好下标 另外有忘记了输入字符时考虑是否有\n */ #include<stdio.h> #include<algorith ...
- PHP 服务器及TP5框架遇到的几个错误
一.Call to undefined function imagecreatefrompng(): LAMP环境搭建的博客,在提交内容的时候TP5框架报了一个错误,Call to undefined ...