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中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

Python Dash框架在数据可视化仪表板中的应用与实践记录

《PythonDash框架在数据可视化仪表板中的应用与实践记录》Python的PlotlyDash库提供了一种简便且强大的方式来构建和展示互动式数据仪表板,本篇文章将深入探讨如何使用Dash设计一... 目录python Dash框架在数据可视化仪表板中的应用与实践1. 什么是Plotly Dash?1.1

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

Java中&和&&以及|和||的区别、应用场景和代码示例

《Java中&和&&以及|和||的区别、应用场景和代码示例》:本文主要介绍Java中的逻辑运算符&、&&、|和||的区别,包括它们在布尔和整数类型上的应用,文中通过代码介绍的非常详细,需要的朋友可... 目录前言1. & 和 &&代码示例2. | 和 ||代码示例3. 为什么要使用 & 和 | 而不是总是使

Python循环缓冲区的应用详解

《Python循环缓冲区的应用详解》循环缓冲区是一个线性缓冲区,逻辑上被视为一个循环的结构,本文主要为大家介绍了Python中循环缓冲区的相关应用,有兴趣的小伙伴可以了解一下... 目录什么是循环缓冲区循环缓冲区的结构python中的循环缓冲区实现运行循环缓冲区循环缓冲区的优势应用案例Python中的实现库

SpringBoot整合MybatisPlus的基本应用指南

《SpringBoot整合MybatisPlus的基本应用指南》MyBatis-Plus,简称MP,是一个MyBatis的增强工具,在MyBatis的基础上只做增强不做改变,下面小编就来和大家介绍一下... 目录一、MyBATisPlus简介二、SpringBoot整合MybatisPlus1、创建数据库和

python中time模块的常用方法及应用详解

《python中time模块的常用方法及应用详解》在Python开发中,时间处理是绕不开的刚需场景,从性能计时到定时任务,从日志记录到数据同步,时间模块始终是开发者最得力的工具之一,本文将通过真实案例... 目录一、时间基石:time.time()典型场景:程序性能分析进阶技巧:结合上下文管理器实现自动计时

Java逻辑运算符之&&、|| 与&、 |的区别及应用

《Java逻辑运算符之&&、||与&、|的区别及应用》:本文主要介绍Java逻辑运算符之&&、||与&、|的区别及应用的相关资料,分别是&&、||与&、|,并探讨了它们在不同应用场景中... 目录前言一、基本概念与运算符介绍二、短路与与非短路与:&& 与 & 的区别1. &&:短路与(AND)2. &:非短

Spring AI集成DeepSeek三步搞定Java智能应用的详细过程

《SpringAI集成DeepSeek三步搞定Java智能应用的详细过程》本文介绍了如何使用SpringAI集成DeepSeek,一个国内顶尖的多模态大模型,SpringAI提供了一套统一的接口,简... 目录DeepSeek 介绍Spring AI 是什么?Spring AI 的主要功能包括1、环境准备2

Spring AI与DeepSeek实战一之快速打造智能对话应用

《SpringAI与DeepSeek实战一之快速打造智能对话应用》本文详细介绍了如何通过SpringAI框架集成DeepSeek大模型,实现普通对话和流式对话功能,步骤包括申请API-KEY、项目搭... 目录一、概述二、申请DeepSeek的API-KEY三、项目搭建3.1. 开发环境要求3.2. mav