Gym101612 Problem G. Grand Test(tarjan,low值应用)

2024-04-16 00:09

本文主要是介绍Gym101612 Problem G. Grand Test(tarjan,low值应用),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题意:
问图中是否存在两个点使得两个点存在至少三条不交叉路径,输出这三条路径。

思路:
被PC拉过来写,写了一个晚上还是不会XXX。
最后参考了题解的写法:记录下每个点的次大low值和最大low值以及对应的点,由此回溯路径就可以保证不交叉了。


本题实际就是找两个环拼在一起的情况。

首先一开始的写法是在点双连通分量里面找到一个度数大于等于3的点,再从这个点出发找到另外一个度数大于等于3的点。因为点双连通分量里面的点一定在环上(排除孤立点),如果度数大于等于3那就说明在不止一个环上,那就一定存在至少三条路径。

再跑dfs找出这两个点所连的三条路径。但是这有个问题,这三条路径确实存在,但是你dfs的过程可能会使得其交叉。用过各种方法(bfs,3条路径同时跑,找最短的先跑等等)。


题解的方法很巧妙,就是维护每个点能到达的最小dfs序low1和次小dfs序low2。
只要存在low2[x]<dfn[x],就说明这个点存在次小能到达dfs序,在跑tarjan的过程中顺带维护每个点上一个节点,再跑回溯就能找到三条不相交路径了。

node存的是对应时序的点,end1[x]代表最小能到达dfs序的点,end2[x]为次小能到达dfs序的点。
起点为x,终点为node[low2[x]]
具体为:

  1. x x x n o d e [ l o w 2 [ x ] ] node[low2[x]] node[low2[x]]
  2. x x x n o d e [ l o w 1 [ x ] ] node[low1[x]] node[low1[x]],再从 n o d e [ l o w 1 [ x ] ] 到 n o d e [ l o w 2 [ x ] ] node[low1[x]]到node[low2[x]] node[low1[x]]node[low2[x]]
  3. x x x e n d 2 [ x ] end2[x] end2[x],再从 e n d 2 [ x ] end2[x] end2[x] n o d e [ l o w 2 [ x ] ] node[low2[x]] node[low2[x]]

总之,这道题加深了我对low值的理解。
只要一个点存在能到达的最小dfs序值(比当前小),那么就说明该点出发能到达之前到过的点,也就是成环。
如果存在次小low值,那就说明存在两条边能到达之前到过的点,那就成了两个环相交。

我们可以再拓展,如果题目要求两点有4条不相交路径,那实际就是3个环拼在一起的情况,再维护次次小low值就好了。
也可以类似的找出这四条路径。

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#include <queue>using namespace std;const int maxn = 1e5 + 7;int dfn[maxn],low1[maxn],low2[maxn],end1[maxn],end2[maxn],node[maxn],cnt;
int head[maxn],nex[maxn * 2],to[maxn * 2],tot;
int Fa[maxn];void add(int x,int y) {to[++tot] = y;nex[tot] = head[x];head[x] = tot;
}void update(int x,int d,int y) {if(d < low1[x]) {low2[x] = low1[x];end2[x] = end1[x];low1[x] = d;end1[x] = y;} else if(d < low2[x]) {low2[x] = d;end2[x] = y;}
}vector<int>get_path(int s,int t,bool flag) { //flag代表是否需要反转vector<int>path;for(int i = s;i != t;i = Fa[i]) {path.push_back(i);}path.push_back(t);if(flag) reverse(path.begin(),path.end());return path;
}void print_path(vector<int>path) {printf("%d ",path.size());for(int i = 0;i < path.size();i++) {printf("%d ",path[i]);}printf("\n");
}int flag = 0;
void tarjan(int x,int fa) {if(flag) return;dfn[x] = low1[x] = low2[x] = ++cnt;node[cnt] = end1[x] = end2[x] = x;for(int i = head[x];i;i = nex[i]) {int v = to[i];if(v == fa) continue;if(!dfn[v]) {Fa[v] = x;tarjan(v,x);update(x,low1[v],end1[v]);} else if(dfn[v] < dfn[x]) {update(x,dfn[v],x);}}if(!flag && low2[x] < dfn[x]) {int s = x,t = node[low2[x]];printf("%d %d\n",s,t);vector<int>path;path = get_path(s,t,false);print_path(path);path = get_path(end2[x],s,true);path.push_back(t);print_path(path);vector<int>path1,path2;path1 = get_path(end1[x],s,true);path2 = get_path(t,node[low1[x]],true);path1.insert(path1.end(),path2.begin(),path2.end());print_path(path1);flag = 1;}
}void init(int n) {tot = 1;cnt = 0;flag = 0;for(int i = 1;i <= n;i++) {low1[i] = low2[i] = end1[i] = end2[i] = node[i] = 0;Fa[i] = dfn[i] = head[i] = 0;}
}int main() {freopen("grand.in","r",stdin);freopen("grand.out","w",stdout);int T;scanf("%d",&T);while(T--) {int n,m;scanf("%d%d",&n,&m);init(n);for(int i = 1;i <= m;i++) {int x,y;scanf("%d%d",&x,&y);add(x,y);add(y,x);}for(int i = 1;i <= n;i++) {if(!dfn[i]) tarjan(i,-1);}if(!flag) {printf("-1\n");}}return 0;
}

这篇关于Gym101612 Problem G. Grand Test(tarjan,low值应用)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/907300

相关文章

5分钟获取deepseek api并搭建简易问答应用

《5分钟获取deepseekapi并搭建简易问答应用》本文主要介绍了5分钟获取deepseekapi并搭建简易问答应用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需... 目录1、获取api2、获取base_url和chat_model3、配置模型参数方法一:终端中临时将加

JavaScript中的isTrusted属性及其应用场景详解

《JavaScript中的isTrusted属性及其应用场景详解》在现代Web开发中,JavaScript是构建交互式应用的核心语言,随着前端技术的不断发展,开发者需要处理越来越多的复杂场景,例如事件... 目录引言一、问题背景二、isTrusted 属性的来源与作用1. isTrusted 的定义2. 为

Python调用另一个py文件并传递参数常见的方法及其应用场景

《Python调用另一个py文件并传递参数常见的方法及其应用场景》:本文主要介绍在Python中调用另一个py文件并传递参数的几种常见方法,包括使用import语句、exec函数、subproce... 目录前言1. 使用import语句1.1 基本用法1.2 导入特定函数1.3 处理文件路径2. 使用ex

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

java中VO PO DTO POJO BO DO对象的应用场景及使用方式

《java中VOPODTOPOJOBODO对象的应用场景及使用方式》文章介绍了Java开发中常用的几种对象类型及其应用场景,包括VO、PO、DTO、POJO、BO和DO等,并通过示例说明了它... 目录Java中VO PO DTO POJO BO DO对象的应用VO (View Object) - 视图对象

Go信号处理如何优雅地关闭你的应用

《Go信号处理如何优雅地关闭你的应用》Go中的优雅关闭机制使得在应用程序接收到终止信号时,能够进行平滑的资源清理,通过使用context来管理goroutine的生命周期,结合signal... 目录1. 什么是信号处理?2. 如何优雅地关闭 Go 应用?3. 代码实现3.1 基本的信号捕获和优雅关闭3.2