1184. 欧拉回路(欧拉回路,模板题)

news/2024/7/15 20:17:02 标签: 图论

活动 - AcWing

给定一张图,请你找出欧拉回路,即在图中找一个环使得每条边都在环上出现恰好一次。

输入格式

第一行包含一个整数 t,t∈{1,2},如果 t=1,表示所给图为无向图,如果 t=2,表示所给图为有向图。

第二行包含两个整数 n,m,表示图的结点数和边数。

接下来 m 行中,第 i 行两个整数 vi,ui,表示第 i 条边(从 11 开始编号)。

  • 如果 t=1 则表示 vi 到 ui 有一条无向边。
  • 如果 t=2 则表示 vi 到 ui 有一条有向边。

图中可能有重边也可能有自环。

点的编号从 1 到 n。

输出格式

如果无法一笔画出欧拉回路,则输出一行:NO。

否则,输出一行:YES,接下来一行输出 任意一组 合法方案即可。

  • 如果 t=1,输出 m 个整数 p1,p2,…,pm。令 e=|pi|,那么 e 表示经过的第 i 条边的编号。如果 pi 为正数表示从 ve 走到 ue,否则表示从 ue 走到 ve。
  • 如果 t=2,输出 m 个整数 p1,p2,…,pm。其中 pi 表示经过的第 i 条边的编号。
数据范围

1≤n≤105
0≤m≤2×105

输入样例1:
1
3 3
1 2
2 3
1 3
输出样例1:
YES
1 2 -3
输入样例2:
2
5 6
2 3
2 5
3 4
1 2
4 2
5 1
输出样例2:
YES
4 1 3 5 2 6

解析: 

一、在无向图中(所有边都是连通的): 

(1)存在欧拉路径的充分必要条件:度数为奇数的点只能有0或2。 

(2)存在欧拉回路(起点和终点相同)的充分必要条件:度数为奇数的点只能有0个。 

二、在有向图中(所有边都是连通的): 

(1)存在欧拉路径的充分必要条件:要么所有点的入度均等于入度;要么除了两个点之外,其余所有的点的出度等于入度,剩余的两个点:一个满足出度比入度多1(起点),另一个满足入度比出度多1(终点)。 

(2)存在欧拉回路(起点和终点相同)的充分必要条件:所有点的入度均等于出度。 

欧拉回路的dfs用边来判重,不能用点。 

#include<iostream>
#include<string>
#include<cstring>
#include<cmath>
#include<ctime>
#include<algorithm>
#include<utility>
#include<stack>
#include<queue>
#include<vector>
#include<set>
#include<math.h>
#include<map>
#include<sstream>
#include<deque>
#include<unordered_map>
#include<unordered_set>
using namespace std;
typedef long long LL;
typedef unsigned long long ULL;
typedef pair<int, int> PII;
const int N = 1e5 + 5, M = 4e5 + 5, INF = 0x3f3f3f3f;

int n, m;
int h[N], e[M], ne[M], idx;
int din[N], dout[N];
int ans[M], cnt;
bool used[M];
int type;

void add(int a, int b) {
	e[idx] = b, ne[idx] = h[a], h[a] = idx++;
}

void dfs(int u) {
	//cout << "_______________________" << u << endl;
	for (int& i = h[u]; i != -1;) {
		if (used[i]) {
			i = ne[i];
			continue;
		}
		int t;
		if (type == 1) {
			t = i / 2 + 1;
			if (i & 1)t = -t;
		}
		else t = i + 1;
		used[i] = 1;
		if (type == 1) {
			used[i ^ 1] = 1;
		}
		int j = e[i];
		i = ne[i];
		dfs(j);
		ans[++cnt] = t;
	}
}

int main() {
	cin >> type;
	cin >> n >> m;
	memset(h, -1, sizeof h);
	for (int i = 1,a,b; i <= m; i++) {
		scanf("%d%d", &a, &b);
		add(a, b);
		if (type == 1)add(b, a);
		din[b]++, dout[a]++;
	}
	if (type == 1) {
		for (int i = 1; i <= n; i++) {
			if (din[i] + dout[i] & 1) {
				cout << "NO" << endl;
				return 0;
			}
		}
	}
	else {
		for (int i = 1; i <= n; i++) {
			if (din[i] != dout[i]) {
				cout << "NO" << endl;
				return 0;
			}
		}
	}
	for (int i = 1; i <= n; i++) {
		if (h[i] != -1) {
			dfs(i);
			break;
		}
	}
	if (cnt < m) {
		cout << "NO" << endl;
		return 0;
	}
	cout << "YES" << endl;
	for (int i = cnt; i; i--) {
		printf("%d ", ans[i]);
	}
	return 0;
}


http://www.niftyadmin.cn/n/5373241.html

相关文章

ts总结1、基本用法

一、类型声明:类型声明的写法,一律为在标识符后面添加“冒号 + 类型” 变量声明: let foo:string = 123; 函数声明: function toString(num:number):string { return String(num); } 含义: num:number标识参数num的类型是number类型; ()string标识函…

问题:3【单选题】实现职业理想的一般步骤是()。 #媒体#媒体

问题&#xff1a;3【单选题】实现职业理想的一般步骤是()。 A、创业-立业-择业 B、择业-创业-立业 C、择业-立业-创业 D、立业-择业-创业 参考答案如图所示

假期2.8

数据类型与作用域练习 1、选择题 1.1、以下选项中,不能作为合法常量的是 ____B______ A&#xff09;1.234e04 B&#xff09;1.234e0.4 C&#xff09;1.234e4 D&#xff09;1.234e0 1.2、以下定义变量并初始化错误的是_____D________。 A) char c1 ‘H’ &a…

对接快团团,自我介绍,寄样,对接话术,最后几个技巧

快团团对接&#xff0c;自我介绍&#xff0c;其实很重要&#xff01; 有些人介绍自己&#xff0c;巴拉巴拉&#xff0c;一大堆&#xff0c;都是无效信息。有些人&#xff0c;只会说你好&#xff0c;你好&#xff0c;然后你让团长怎么回&#xff1f;有些人&#xff0c;加上微信&…

龙芯+RT-Thread+LVGL实战笔记(34)——密码锁初步

【写在前面】春节期间,本系列教程会不定期更新,毕竟这是一年当中最适合放下工作的时期。祝各位朋友新年大吉,身体健康。来年继续关注笔者和CSDN平台,收获更多的知识和技能。按照惯例,还是在开篇做一些声明: 有些硬件模块笔者并没有,如LED点阵、压力传感模块、RFID模块等…

软件价值12-射箭游戏

射箭游戏&#xff0c;按空格键发射&#xff0c;打击移动靶&#xff0c;左上角显示成绩状态。 代码&#xff1a; import pygame import sys import random# 初始化Pygame pygame.init()# 设置窗口大小 SCREEN_WIDTH 800 SCREEN_HEIGHT 600 screen pygame.display.set_mode((…

C语言什么是悬空指针?

一、问题 什么是悬空指针&#xff1f;为什么会出现&#xff1f;我们该如何避免悬空指针的出现&#xff1f; 二、解答 在C语言中&#xff0c;悬空指针指的是指向已删除&#xff08;或释放&#xff09;的内存位置的指针。如果一个指针指向的内存被释放&#xff0c;但指针本身并未…

【算法与数据结构】42、LeetCode接雨水

文章目录 一、题目二、解法三、完整代码 所有的LeetCode题解索引&#xff0c;可以看这篇文章——【算法和数据结构】LeetCode题解。 一、题目 二、解法 思路分析&#xff1a;   程序如下&#xff1a; 复杂度分析&#xff1a; 时间复杂度&#xff1a; O ( ) O() O()。空间复…