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

相关文章

将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

正则表达式高级应用与性能优化记录

《正则表达式高级应用与性能优化记录》本文介绍了正则表达式的高级应用和性能优化技巧,包括文本拆分、合并、XML/HTML解析、数据分析、以及性能优化方法,通过这些技巧,可以更高效地利用正则表达式进行复杂... 目录第6章:正则表达式的高级应用6.1 模式匹配与文本处理6.1.1 文本拆分6.1.2 文本合并6

python中的与时间相关的模块应用场景分析

《python中的与时间相关的模块应用场景分析》本文介绍了Python中与时间相关的几个重要模块:`time`、`datetime`、`calendar`、`timeit`、`pytz`和`dateu... 目录1. time 模块2. datetime 模块3. calendar 模块4. timeit

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取