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

相关文章

Java中的Lambda表达式及其应用小结

《Java中的Lambda表达式及其应用小结》Java中的Lambda表达式是一项极具创新性的特性,它使得Java代码更加简洁和高效,尤其是在集合操作和并行处理方面,:本文主要介绍Java中的La... 目录前言1. 什么是Lambda表达式?2. Lambda表达式的基本语法例子1:最简单的Lambda表

Python结合PyWebView库打造跨平台桌面应用

《Python结合PyWebView库打造跨平台桌面应用》随着Web技术的发展,将HTML/CSS/JavaScript与Python结合构建桌面应用成为可能,本文将系统讲解如何使用PyWebView... 目录一、技术原理与优势分析1.1 架构原理1.2 核心优势二、开发环境搭建2.1 安装依赖2.2 验

Java字符串操作技巧之语法、示例与应用场景分析

《Java字符串操作技巧之语法、示例与应用场景分析》在Java算法题和日常开发中,字符串处理是必备的核心技能,本文全面梳理Java中字符串的常用操作语法,结合代码示例、应用场景和避坑指南,可快速掌握字... 目录引言1. 基础操作1.1 创建字符串1.2 获取长度1.3 访问字符2. 字符串处理2.1 子字

SpringShell命令行之交互式Shell应用开发方式

《SpringShell命令行之交互式Shell应用开发方式》本文将深入探讨SpringShell的核心特性、实现方式及应用场景,帮助开发者掌握这一强大工具,具有很好的参考价值,希望对大家有所帮助,如... 目录引言一、Spring Shell概述二、创建命令类三、命令参数处理四、命令分组与帮助系统五、自定

SpringBoot应用中出现的Full GC问题的场景与解决

《SpringBoot应用中出现的FullGC问题的场景与解决》这篇文章主要为大家详细介绍了SpringBoot应用中出现的FullGC问题的场景与解决方法,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录Full GC的原理与触发条件原理触发条件对Spring Boot应用的影响示例代码优化建议结论F

MySQL 分区与分库分表策略应用小结

《MySQL分区与分库分表策略应用小结》在大数据量、复杂查询和高并发的应用场景下,单一数据库往往难以满足性能和扩展性的要求,本文将详细介绍这两种策略的基本概念、实现方法及优缺点,并通过实际案例展示如... 目录mysql 分区与分库分表策略1. 数据库水平拆分的背景2. MySQL 分区策略2.1 分区概念

Spring Shell 命令行实现交互式Shell应用开发

《SpringShell命令行实现交互式Shell应用开发》本文主要介绍了SpringShell命令行实现交互式Shell应用开发,能够帮助开发者快速构建功能丰富的命令行应用程序,具有一定的参考价... 目录引言一、Spring Shell概述二、创建命令类三、命令参数处理四、命令分组与帮助系统五、自定义S

C语言函数递归实际应用举例详解

《C语言函数递归实际应用举例详解》程序调用自身的编程技巧称为递归,递归做为一种算法在程序设计语言中广泛应用,:本文主要介绍C语言函数递归实际应用举例的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录前言一、递归的概念与思想二、递归的限制条件 三、递归的实际应用举例(一)求 n 的阶乘(二)顺序打印

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

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

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