割点与其例题

割点

定义:

若一个点在图中被去掉后,图的连通块个数增加,那么这个点就被称为“割点”。如下图所示红点。

定义说白了就是若去掉一个点,图被“断开”的点称为割点。

朴素算法:

  • 枚举每个点 u。
  • 遍历图,如果有一个点或多个点遍历不到(遍历期间不能经过点 u),那么 u 就是割点。

时间复杂度: O ( N 2 ) O(N^2) O(N2)

可作为对拍暴力程序。

正解:Tarjan

定义一些东西:

  1. 时间戳:dfs 时表示每个点被遍历到的“时间”,可用一个不断增加的变量实现。记为 d f n dfn dfn
  2. 搜索树:dfs 时由遍历到的边组成的树(由于有打标记,所以不会重复访问)。
  3. 追溯值:以 u 为根的子树中,所有不经过 u 能够到达的节点的时间戳的最小值。记为 l o w low low
关于追溯值:

结合张图来理解:

设红边为搜索树的边,则 3 3 3 号点因为有蓝色的边不经过他的父亲 2 2 2 号点,直接到达了 1 1 1 号点,所以 l o w 3 = d f n 1 low_3=dfn_1 low3=dfn1

回归 Tarjan

有一个重要的概念:

一个点 u 如果是割点,那么它的子树中的一些节点 v 的 l o w v low_v lowv 是大等于 d f n u dfn_u dfnu 的,因为它到不了上面(上面的意思是搜索树中比 u 更早遍历到的点集)。

显然, l o w u low_u lowu 表示假设断开点 u 孩子们还能遍历到的最早时间戳。

l o w v ≥ d f n u low_v\ge dfn_u lowvdfnu (v 是 u 的孩子),即 v 回不到 u 前,那么就表示 u 是割点。

s s s 个这样的 v 就代表断开 u 可以把原先的连通图变成 s + 1 s+1 s+1 个连通块(u 上方也是一个)。

遍历路上

对于每个点 u,遍历到的儿子 v 有两种可能:

  1. d f n v = 0 dfn_v=0 dfnv=0

说明 v 是新加入搜索树中的节点,那么就先递归下去,用 l o w v low_v lowv 更新 l o w u low_u lowu

l o w u = m i n ( l o w u , l o w v ) low_u=min(low_u,low_v) lowu=min(lowu,lowv)

  1. d f n v ≠ 0 dfn_v\neq 0 dfnv=0

说明 v 曾经

被遍历过,是搜索树上 u 的祖先,那么用 d f n v dfn_v dfnv 更新 l o w u low_u lowu

l o w u = m i n ( l o w u , d f n v ) low_u=min(low_u,dfn_v) lowu=min(lowu,dfnv)

然而上述办法还是有 bug。想想在哪呢?

发现 bug

假设我们搜索树从 1 1 1 号点开始遍历,给张图你就懂。

如图。

因为我们是从 1 1 1 号点开始遍历的, 1 1 1 号点是搜索树的根,它哪来的祖先能让孩子们去更新追溯值啊!!!

而图中的 1 1 1 号点又显然不是割点。

咋办呢?

解决 bug

特判呗。反正根只有一个。

这时候我们得思考:什么样的情况下根是割点?

反正追溯值做不了了。

那么看看朴素的图吧。

图中 1 1 1 号点就是割点。

为啥嘞?

答:因为把它删了后有两个连通块。

正解。

我们记录一下,如果它在搜索树上的儿子不止一个,那么它就是割点。

就这么简单?

就这么简单。

这时候不知道有没有同学有个疑惑和我初学时一样的,如图:

红色的是搜索树边。

图中 1 1 1 不是割点啊,但它在树上还真有两个孩子啊??

如果您一开始没看出来哪儿错了,就点个赞再走吧。

注意到边 3 → 2 3\rightarrow 2 32 1 → 2 1\rightarrow 2 12

当我们遍历到点 3 3 3 的时候,它就会顺带把 2 2 2 号点先遍历了。先遍历到 2 2 2 再遍历 3 3 3 同理。

所以说搜索树应该为:

或:

OK,下班,看题。

洛谷 P3388 【模板】割点(割顶)。

题意很简略了。就是看看实现。

#include<bits/stdc++.h>
using namespace std;
const int N=2e4+5,M=1e5+5;
int n,m,ehead[N],cnt_e,low[N],dfn[N],idx,rt,cntans;
bool ans[N];//是否为割点
struct E{int to,pre;
}e[M<<1];
void adde(int from,int to)
{e[++cnt_e].to=to;e[cnt_e].pre=ehead[from];ehead[from]=cnt_e;return;
}
void dfs(int u)
{low[u]=dfn[u]=++idx;int chtree=0;//如果是根的话,它的孩子个数for(int i=ehead[u];i;i=e[i].pre){int v=e[i].to;if(!dfn[v])//不在搜索树上{dfs(v);low[u]=min(low[u],low[v]);if(rt==u)++chtree;if(low[v]>=dfn[u]&&rt!=u&&(!ans[u]))//注意 (!ans[u])。搞不好会重复算 cntans{++cntans;ans[u]=1;}}else//返祖边low[u]=min(low[u],dfn[v]);}if(u==rt&&chtree>1&&(!ans[u])){++cntans;ans[u]=1;}return;
}
int main(){ios::sync_with_stdio(0);cin>>n>>m;for(int i=1,u,v;i<=m;++i){cin>>u>>v;adde(u,v);adde(v,u);}for(int i=1;i<=n;++i)//图不保证联通{if(!dfn[i]){rt=i;dfs(i);}}cout<<cntans<<'\n';for(int i=1;i<=n;++i)if(ans[i])cout<<i<<' ';cout<<'\n';return 0;
}

闲话时间

讲个好玩的,这篇文章是我晚上十一点左右写的,但是:

我来自报家门了。

正题。

Tarjan 算法不光能解决割点的问题,改一改还能当作强连通分量和割边(又称桥)和双连通分量等等。

说到强连通分量,推销一下我的学习笔记不过分吧 qwq。

完结撒花。

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

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

相关文章

图卷积神经网络(Graph Convolutional Network, GCN)

最近看论文看到了图卷积神经网络的内容&#xff0c;之前整理过图神经网络的内容&#xff0c;这里再补充一下&#xff0c;方便以后查阅。 图卷积神经网络&#xff08;Graph Convolutional Network, GCN&#xff09; 图卷积神经网络1. 什么是图卷积神经网络&#xff08;GCN&#…

安装win11硬盘分区MBR还是GPT_装win11系统分区及安装教程

最近有网友问我,装win11系统分区有什么要求装win11系统硬盘分区用mbr还是GPT&#xff1f;我们知道现在的引导模式有uefi和legacy两种引导模式&#xff0c;如果采用的是uefi引导模式&#xff0c;分区类型对应的就是gpt分区(guid)&#xff0c;如果引导模式采用的是legacy&#xf…

服务培训QDA 的安装调试方法,硬件模块的讲解和软件控制台使用及系统测试

#服务培训##质谱仪##软件控制##硬件模块# 以下是关于Waters QDa单杆液质质谱仪的安装调试、硬件模块讲解以及软件控制台使用培训的相关内容&#xff1a; 安装调试 场地准备&#xff1a;用户需要提前准备好实验室&#xff0c;确保实验室环境符合仪器的要求&#xff0c;如温度、…

在K8S集群中部署EFK日志收集

目录 引言环境准备安装自定义资源部署ElasticsearchMaster 节点与 Data 节点的区别生产优化建议安装好以后测试ES是否正常部署Fluentd测试filebeat是否正常推送日志部署Kibana获取账号密码&#xff0c;账号是&#xff1a;elastic集群测试 引言 系统版本为 Centos7.9内核版本为…

polarctf-web-[rce1]

考点&#xff1a; (1)RCE(exec函数) (2)空格绕过 (3)执行函数(exec函数) (4)闭合(ping命令闭合) 题目来源&#xff1a;Polarctf-web-[rce1] 解题&#xff1a; 这段代码实现了一个简单的 Ping 测试工具&#xff0c;用户可以通过表单提交一个 IP 地址&#xff0c;服务器会执…

【串流VR手势】Pico 4 Ultra Enterprise 在 SteamVR 企业串流中无法识别手势的问题排查与解决过程(Pico4UE串流手势问题)

写在前面的话 此前&#xff08;用Pico 4U&#xff09;接入了MRTK3&#xff0c;现项目落地需要部署&#xff0c;发现串流场景中&#xff0c;Pico4UE的企业串流无法正常识别手势。&#xff08;一体机方式部署使用无问题&#xff09; 花了半小时解决&#xff0c;怕忘&#xff0c;…

ES(Elasticsearch)的应用与代码示例

Elasticsearch应用与代码示例技术文章大纲 一、引言 Elasticsearch在现代化应用中的核心作用典型应用场景分析&#xff08;日志分析/全文检索/数据聚合&#xff09; 二、环境准备(前提条件) Elasticsearch 8.x集群部署要点IK中文分词插件配置指南Ingest Attachment插件安装…

临床决策支持系统的提示工程优化路径深度解析

引言 随着人工智能技术在医疗领域的迅猛发展,临床决策支持系统(CDSS)正经历从传统规则引擎向智能提示工程的范式转变。在这一背景下,如何构建既符合循证医学原则又能适应个体化医疗需求的CDSS成为医学人工智能领域的核心挑战。本报告深入剖析了临床决策支持系统中提示工程的…

火山RTC 8 SDK集成进项目中

一、SDK 集成预备工作 1、SDK下载 https://www.volcengine.com/docs/6348/75707 2、解压后 3、放在自己项目中的位置 1&#xff09;、include 2&#xff09;、lib 3)、dll 暂时&#xff0c;只需要VolcEngineRTC.dll RTCFFmpeg.dll openh264-4.dll&#xff0c; 放在intLive2…

OkHttp用法-Java调用http服务

特点&#xff1a;高性能&#xff0c;支持异步请求&#xff0c;连接池优化 官方文档&#xff1a;提供快速入门指南和高级功能&#xff08;如拦截器、连接池&#xff09;的详细说明&#xff0c;GitHub仓库包含丰富示例。 社区资源&#xff1a;中文教程丰富&#xff0c;GitHub高…

python中常用的参数以及命名规范

以下是 Python 中常见的命名规范、参数用法及在大型项目中常用的操作模式&#xff0c;供记录参考&#xff1a; 1. 命名规范&#xff08;Naming Conventions&#xff09; 前缀/形式含义示例_age单下划线&#xff1a;弱“私有”标记&#xff08;可访问但不建议外部使用&#xff…

第五十七篇 Java接口设计之道:从咖啡机到智能家居的编程哲学

目录 引言&#xff1a;生活中的接口无处不在一、咖啡机与基础接口&#xff1a;理解抽象契约1.1 咖啡制作的标准接口 二、智能家居与策略模式&#xff1a;灵活切换实现2.1 温度调节策略场景 三、物流系统与工厂模式&#xff1a;标准接口下的多样实现3.1 快递运输接口设计 四、健…

第二十六天打卡

全局变量 global_var 全局变量是定义在函数、类或者代码块外部的变量&#xff0c;它在整个程序文件内都能被访问。在代码里&#xff0c; global_var 就是一个全局变量&#xff0c;下面是相关代码片段&#xff1a; print("\n--- 变量作用域示例 ---") global_var …

联合查询

目录 1、笛卡尔积 2、联合查询 2.1、内连接 2.2、外连接 1、笛卡尔积 笛卡尔积&#xff1a; 笛卡尔积是让两个表通过排列组合的方式&#xff0c;得到的一个更大的表。笛卡尔积的列数&#xff0c;是这两个表的列数相加&#xff0c;笛卡尔积的行数&#xff0c;是这两个表的行…

【HTML5学习笔记2】html标签(下)

1表格标签 1.1表格作用 显示数据 1.2基本语法 <table><tr> 一行<td>单元格1</td></tr> </table> 1.3表头单元格标签 表头单元格会加粗并且居中 <table><tr> 一行<th>单元格1</th></tr> </table&g…

window 显示驱动开发-分页视频内存资源

与 Microsoft Windows 2000 显示驱动程序模型不同&#xff0c;Windows Vista 显示驱动程序模型允许创建比可用物理视频内存总量更多的视频内存资源&#xff0c;然后根据需要分页进出视频内存。 换句话说&#xff0c;并非所有视频内存资源都同时位于视频内存中。 GPU 的管道中可…

《C 语言指针高级指南:字符、数组、函数指针的进阶攻略》

目录 一. 字符指针变量 二. 数组指针变量 三. 二维数组传参 3.1 二维数组的本质 3.2 访问方式与地址计算 3.3 二维数组的传参方式 3.4 深入解析 *(*(arri)j) 与 arr[i][j] 的等价性 四. 函数指针变量 4.1 函数指针变量的创建 4.2 函数指针变量的使用 4.3 两段"…

Unity:场景管理系统 —— SceneManagement 模块

目录 &#x1f3ac; 什么是 Scene&#xff08;场景&#xff09;&#xff1f; Unity 项目中的 Scene 通常负责什么&#xff1f; &#x1f30d; 一个 Scene 包含哪些元素&#xff1f; Scene 的切换与管理 &#x1f4c1; 如何创建与管理 Scenes&#xff1f; 什么是Scene Man…

内容中台重构企业知识管理路径

智能元数据驱动知识治理 现代企业知识管理的核心挑战在于海量非结构化数据的有效治理。通过智能元数据分类引擎&#xff0c;系统可自动识别文档属性并生成多维标签体系&#xff0c;例如将技术手册按产品版本、功能模块、适用场景进行动态标注。这种动态元数据框架不仅支持跨部…

Vue3:脚手架

工程环境配置 1.安装nodejs 这里我已经安装过了&#xff0c;只需要打开链接Node.js — Run JavaScript Everywhere直接下载nodejs&#xff0c;安装直接一直下一步下一步 安装完成之后我们来使用电脑的命令行窗口检查一下版本 查看npm源 这里npm源的地址是淘宝的源&#xff0…