给定n个结点m条边的简单无向图,判断该图是否存在鱼形状的子图:有一个环,其中有一个结点有另外两条边,连向不在环内的两个结点。若有,输出子图的连边

题目

思路:

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
#define int long long
#define pb push_back
#define fi first
#define se second
#define lson p << 1
#define rson p << 1 | 1
const int maxn = 1e6 + 5, inf = 1e18 * 3, maxm = 4e4 + 5, mod = 1e9 + 7;
int a[maxn], b[maxn];
int n, m;
string s;
bool vis[maxn];
vector<int> G[maxn];
int from, to;
vector<int> tmp, vec;
int deg[maxn];

void dfs(int u, int p){
	if(u == to && p == from) return;
	if(vis[u]) return;
	vis[u] = 1;
	tmp.pb(u);
	if(u == to){
		vec = tmp;
		return;
	}
	for(auto v : G[u]){
		dfs(v, u);
	}
	tmp.pop_back();
}
void solve(){
    int res = 0;
    int k, x, q;
	cin >> n >> m;
	for(int i = 1; i <= n; i++){
		G[i].clear();
		deg[i] = 0;
	}
	for(int i = 1; i <= m; i++){
		int u, v;
		cin >> u >> v;
		G[u].pb(v);
		G[v].pb(u);
		deg[u]++;
		deg[v]++;
	}
	for(int u = 1; u <= n; u++){
		if(deg[u] < 4) continue;
		for(auto v : G[u]){
			for(int i = 1; i <= n; i++){
				vis[i] = 0;
			}
			from = u;
			to = v;
			vec.clear();
			tmp.clear();
			dfs(u, u);
			if(vec.empty()) continue;
			vector<int> extra;
			tmp.clear();
			for(auto x : G[u]){
				if(x == vec.back() || x == *(vec.begin() + 1)) continue;
				if(find(vec.begin(), vec.end(), x) == vec.end()){//不在环内
					extra.pb(x);
					if(extra.size() == 2) break;
				}
				else{
					tmp.pb(x);//在环内,且与u相连
				}
			}
			vector<int> vt;
			if(extra.size() < 2){
				for(auto x : vec){
					vt.pb(x);
					if(find(tmp.begin(), tmp.end(), x) != tmp.end()){//在tmp内
						break;
					}
				}
				
				extra.pb(vec.back());
				for(int i = vec.size() - 1; i >= 0; i--){
					int x = vec[i];
					if(find(tmp.begin(), tmp.end(), x) != tmp.end()){
						extra.pb(x);
						break;
					}
				}
				// extra.pb(vec.end() - 2) 不能这样写,因为这个结点不一定与u相连
				// extra.pb(tmp.back()); 不能这样写,因为tmp的顺序跟环的结点顺序不一致

				extra.resize(2);
				vec = vt;
			}
			cout << "Yes\n";
			cout << vec.size() + 2 << '\n';
			int m = vec.size();
			for(int i = 0; i < vec.size(); i++){
				cout << vec[i] << ' ' << vec[(i + 1) % m] << '\n';
			}
			cout << u << ' ' << extra[0] << '\n';
			cout << u << ' ' << extra[1] << '\n';
			return;
		}
		
	}
	cout << "No\n";
}
    
signed main(){
    ios::sync_with_stdio(0);
    cin.tie(0);
    int T = 1;
    cin >> T;
    while (T--)
    {
        solve();
    }
    return 0;
}
相关推荐
888CC++4 小时前
如何在 C 语言中进行程序调试?
前端·javascript·算法
(●—●)橘子……5 小时前
力扣第503场周赛练习理解
python·学习·算法·leetcode·职场和发展·周赛
明志数科7 小时前
4D时序标注技术详解:让机器人理解连续动作的数据基础
java·算法·机器人
KaMeidebaby7 小时前
卡梅德生物技术快报|原核表达系统工艺优化:包涵体重折叠 + 分子筛纯化实现功能 RBD 高效制备,附全参数配置
前端·人工智能·算法·数据挖掘·数据分析
无限码力7 小时前
携程0510笔试真题【单数组交换】
算法·携程笔试·携程笔试真题·携程0510笔试真题
BlockWay8 小时前
WEEX Labs 周度观察:微软-OpenAI 合作调整与AI 多云趋势
大数据·人工智能·算法·安全·microsoft
风筝在晴天搁浅8 小时前
快手 CodeTop LeetCode 224.基本计算器
数据结构·算法·leetcode
Smoothcloud润云8 小时前
5大功能精修,重构AI算力使用体验!
java·人工智能·windows·算法·重构·编辑器·sublime text
计算机安禾9 小时前
【算法分析与设计】第41篇:确定性与非确定性多项式时间:P与NP的形式化
算法