图论算法<三>:判断有向图中是否有存在循环 ,以及环的个数和各个环中的元素

2023-11-18

1、目的

  判断有向图中是否有存在循环 ,以及环的个数和各个环中的元素。

2、示例效果

2.1 原始数据

在这里插入图片描述
路线起终点整理如下:

// 共计12个顶点,19条边。 (起点,终点,1)最后的1代表起点终点是连通的。
起点,终点,12 4 1
起点,终点,19 10 1
起点,终点,18 11 1
起点,终点,14 12 1
起点,终点,111 12 1
起点,终点,11 2 1
起点,终点,13 2 1
起点,终点,11 3 1
起点,终点,13 4 1
起点,终点,13 6 1
起点,终点,11 5 1
起点,终点,16 5 1
起点,终点,16 7 1
起点,终点,16 9 1
起点,终点,17 9 1
起点,终点,19 10 1
起点,终点,15 8 1
起点,终点,18 7 1
起点,终点,110 11 1

为方便测试:将分支的起终点互换改为:e13:7-6; e3:11-8.

2.2 程序计算效果

在这里插入图片描述

3、算法源码

// CheckGraphCircle.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//
#include <iostream>
#include <vector>
using namespace std;
const int MAX_Vertex_Num = 500;
template<class VexType, class ArcType>
class MGraph
{
public:
	//创建图
	void CreateGraph();//创建图

	//返回顶点v所在顶点向量中的位置(下标)
	int LocateVex(VexType v);

	//检查图中有无环
	bool CheckCircle(int& nCount,std::vector<vector<int>>& vecCircle);

private:
	VexType m_vexs[MAX_Vertex_Num];//顶点向量
	ArcType m_arcs[MAX_Vertex_Num][MAX_Vertex_Num]; //这里把邻接矩阵类型用模板表示,主要是为了处理有权值的情况,比如:权值可以为小数,也可以为整数
	int m_vexNum;//顶点数
	int m_arcNum;//边数

private:
	void DFS(int x, bool visited[MAX_Vertex_Num], int stack[MAX_Vertex_Num], int& top, bool inStack[MAX_Vertex_Num], int& count, std::vector<vector<int>>& vecCircle);
};

template<class VexType, class ArcType>
void MGraph<VexType, ArcType>::CreateGraph()
{
	//VexType first;
	//VexType Secend;
	//cout << "请输入顶点数:";
	//cin >> vexnum;
	//cout << "请输入边数:";
	//cin >> arcnum;

	m_vexNum = 12;
	cout << "顶点数: 12" << endl;
	m_arcNum = 19;
	cout << "边数: 19" << endl;

	//cout << "请输入各个顶点值:";
	for (int i = 1; i <= m_vexNum; i++)
	{
		//cin >> vexs[i];
		m_vexs[i] = i;
	}

	//初始化邻接矩阵
	for (int i = 0; i < m_arcNum; i++)
	{
		for (int j = 0; j < m_arcNum; j++)
		{
			m_arcs[i][j] = 0;
		}
	}

	// 为边赋值
	m_arcs[2][4] = 1;

	m_arcs[9][10] = 1;

	m_arcs[11][8] = 1;

	m_arcs[4][12] = 1;

	m_arcs[11][12] = 1;

	m_arcs[1][2] = 1;

	m_arcs[3][2] = 1;

	m_arcs[1][3] = 1;

	m_arcs[3][4] = 1;

	m_arcs[3][6] = 1;

	m_arcs[1][5] = 1;

	m_arcs[6][5] = 1;

	m_arcs[7][6] = 1;

	m_arcs[6][9] = 1;

	m_arcs[7][9] = 1;

	m_arcs[9][10] = 1;

	m_arcs[5][8] = 1;

	m_arcs[8][7] = 1;

	m_arcs[10][11] = 1;

	//cout << "请输入边的信息:" << endl;
	//for (int i = 0; i < arcnum; i++)
	//{
	//	cin >> first >> Secend;

	//	//如果边有权值的话,则还应该输入权值
	//	int x = LocateVex(first);
	//	int y = LocateVex(Secend);

	//	arcs[x][y] = 1;//如果是有权的话,这里应该是arc[x][y]=权值
	//}
}

/*
参数:v:表示顶点向量中一个值
函数返回值:函数返回v在顶点向量中的下标
*/
template<class VexType, class ArcType>
int MGraph<VexType, ArcType>::LocateVex(VexType v)
{
	for (int i = 0; i < m_vexNum; i++)
	{
		if (m_vexs[i] == v)
		{
			return i;
		}
	}
	return -1;
}

/*
检查图中是不是有回向边
思想:
如果有回向边,则无环,反之有环
*/
template<class VexType, class ArcType>
bool MGraph<VexType, ArcType>::CheckCircle(int& nCount, std::vector<vector<int>>& vecCircle)
{
	nCount = 0;//环的个数
	int top = -1;
	int stack[MAX_Vertex_Num];
	bool inStack[MAX_Vertex_Num] = { false };
	bool visited[MAX_Vertex_Num] = { false };
	for (int i = 0; i < m_vexNum; i++)
	{
		if (!visited[i])
		{
			DFS(i, visited, stack, top, inStack, nCount, vecCircle);
		}
	}

	return (nCount > 0) ? true : false;
}

bool CompareVector(vector<int> vec1, vector<int> vec2)
{
	if (vec1.size() != vec2.size())
	{
		return false;
	}

	if(vec1.size() == 0)
	{
		return true;
	}

	int nFirstValue = vec1[0];
	int nIndex = -1;
	for (int i = 0; i < vec2.size(); ++i)
	{
		if (vec2[i] == nFirstValue)
		{
			nIndex = i;
			break;
		}
	}

	int nCounter = 0;
	vector<int> vecMid;
	for (int i = 0; i < vec2.size(); ++i)
	{
		if (nIndex > vec2.size() - 1)
		{
			nIndex = 0;
		}
		vecMid.push_back(vec2[nIndex]);
		nIndex++;
	}

	for (int i = 0; i < vec1.size(); ++i)
	{
		if (vec1[i] != vecMid[i])
		{
			return false;
		}
	}

	return true;
}

template<class VexType, class ArcType>
void MGraph<VexType, ArcType>::DFS(int x, bool visited[MAX_Vertex_Num], int stack[MAX_Vertex_Num], int& top, bool inStack[MAX_Vertex_Num], int& nCount, std::vector<vector<int>>& vecCircle)
{
	visited[x] = true;
	stack[++top] = x;
	inStack[x] = true;
	for (int i = 0; i < m_vexNum; i++)
	{
		if (m_arcs[x][i] != 0)//有边
		{
			if (!inStack[i])
			{
				DFS(i, visited, stack, top, inStack, nCount, vecCircle);
			}
			else //条件成立,表示下标为x的顶点到 下标为i的顶点有环
			{
				nCount++;
				//cout << "第"<< nCount << "环为:";
				//从i到x是一个环,top的位置是x,下标为i的顶点在栈中的位置要寻找一下
				//寻找起始顶点下标在栈中的位置
				int t = 0;
				for (t = top; stack[t] != i; t--);
				//输出环中顶点
				vector<int> vecNode;
				for (int j = t; j <= top; j++)
				{
					//cout << (int)m_vexs[stack[j]];
					vecNode.push_back((int)m_vexs[stack[j]]);
				}
				//cout << endl;

				bool bHas = false;
				for (int i = 0; i < vecCircle.size(); i++)
				{
					vector<int> vecTemp = vecCircle[i];
					if (CompareVector(vecTemp, vecNode))
					{
						bHas = true;
						break;
					}
				}

				if (!bHas)
				{
					vecCircle.push_back(vecNode);
				}
			}
		}
	}
	//处理完结点后,退栈
	top--;
	inStack[x] = false;
}

int main()
{
	MGraph<char, int> myGraph;
	myGraph.CreateGraph();

	int nCount = 0;
	std::vector<vector<int>> vecCircle;
	if (myGraph.CheckCircle(nCount, vecCircle))
	{
		cout << "当前图中存在局部循环个数:" << vecCircle.size() << endl;
	}
	else
	{
		cout << "当前图中不存在局部循环!" << endl;
	}

	for (int i = 0; i < vecCircle.size(); i++)
	{
		cout << "第" << i << "个环为:";

		vector<int> vecTemp = vecCircle[i];
		for (int j = 0; j < vecTemp.size(); j++)
		{
			cout << vecTemp[j] << " ";
		}

		cout << endl;
	}
	
	return 1;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

图论算法<三>:判断有向图中是否有存在循环 ,以及环的个数和各个环中的元素 的相关文章

  • Java封装性(包含this关键字,构造器等)

    目录 一 封装性的含义 二 封装性的作用 三 封装性的体现 3 1 四种权限修饰符的介绍 3 2 分装性具体的实现 四 构造器的解释 4 1 构造器的作用 4 2 注意事项 4 3 构造器的举例说明 五 this关键字的使用 5 1 thi

随机推荐

  • jdk8的十大新特性

    了解jdk8的新特性首先要了解jdk7的新特新都有哪些方面增强 1 jdk语法上 1 1二进制变量的表示 支持将整数类型用二进制来表示 用Ob开头 1 2Switch语句支持String类型 1 3Try with resource语句 注
  • 欢欢喜喜: 在lenovo网站购T61的经历

    1月3日 在lenovo网站购T61的经历 一直以来对IBM的小黑情有独钟 不过考虑国内昂贵的价格和需求的迫切性不高 所以 也只是观望中 上次去米果的时候 看到lenovo网上卖的T61笔记本 标的价格比平时都低300 于是动了心 开始在
  • 使用 PullToRefresh 的总结

    前言 关于下拉刷新 上拉加载的框架现在有很多 这里奉上别人收集的一些框架 下拉刷新框架收集 但是笔者一直还在使用 PullToRefresh 个人觉得 PullToRefresh 使用起来还是比较简洁方便的 关于 PullToRefresh
  • unity-ugui-eventsystem

    EventSystem对象的说明 当我们在场景中创建任一UI对象后 Hierarchy面板中都可以看到系统自动创建了对象EventSystem 可以看到该对象下有三个组件 EventSystem StandaloneInputModule
  • 七天引爆社交新零售(助你提高十倍业绩)——前言

    2019年对于中小企业主 创业者 实体店主最大的机遇就是社交新零售 为什么这么说呢 随着日益上涨房租成本 人工成本的上升 实体生意利润空间越来越小了 而传统线上电商企业流量广告费也越来越贵了 大家一直在探索有没有一种低成本高收益的销售方式出
  • stream()转map转list、distinct()去重、判断空值、sorted排序正序多字段排序

    package demo io import demo api JavaBean Student import org junit platform commons util StringUtils import java util imp
  • 镜头快速精准反馈位置硬件环境搭建

    目录 概述 一 检测部分 1 原理图 2 PCB板 二 驱动部分 1 原理图 2 PCB板 概述 本篇只要介绍 硬件电路搭建 这次是项目的需要 重新捡起好多年没使用 Altium Designer 软件了 熟悉又陌生 经过2 3天时间 终于
  • 想要精通算法和SQL的成长之路 - 最长回文子序列

    想要精通算法和SQL的成长之路 最长回文子序列 前言 一 最长回文子序列 前言 想要精通算法和SQL的成长之路 系列导航 一 最长回文子序列 原题链接 首先 我们看下动态规划方程的定义 我们用dp i j 来代表 字符串s在下标区间为 i
  • Django连接MySQL数据库时出错:django.core.exceptions.ImproperlyConfigured: Error loading MySQLdb module: No mo

    基于python3解释器的虚拟环境中创建的Django项目 Django中默认连接的是SQLite3数据库 现更改为MySQL数据库 执行迁移文件时报错 django core exceptions ImproperlyConfigured
  • html5 调用本地街景,H5案例分享:在移动端调用腾讯街景

    在移动端调用腾讯街景 腾讯地图街景组件可以通过多种方式调起 来展示3D街景信息 腾讯街景API 是构建在v2版本上的全新应用接口 对于目的地 可以让用户足不出户 得到更直观 更真切 的身临其境的体验 比如 您可以就用在 房产 酒店 餐饮 娱
  • 使用Java 8函数式编程生成字母序列

    在 Java 8 中使用函数式编程生成字母序列是一个很大的挑战 Lukas Eder 愉快地接受了这个挑战 他将告诉我们如何使用 Java 8 来生成ABC的序列 当然 肯定不是一种蹩脚的方式 我被 Stack Overflow 上网友 m
  • C++ xml库的选择

    自从触及xml文件的读写 一直以来都是用的tinyxml2 接口简单 然而近期项目频繁出错 跟踪调试发现 问题出在了xml文件的读写上 当节点数超过百万级别的时候 内存暴增到G的当量 很显然程序会由于内存申请不足崩掉了 果断寻找替代品 百度
  • Android开源框架之Picasso(图片加载框架)

    简介 Picasso是Square公司出品的一个强大的图片下载和缓存图片库 在adapter中需要取消已经不在视野范围的ImageView图片资源的加载 否则会导致图片错位 Picasso已经解决了这个问题 使用复杂的图片压缩转换来减少内存
  • ue4 蓝图通信的几种方式

    一 设置公有变量 完成通信 1 蓝图类Door bp中声明变量NewVar 1 为公有 确定好变量类型 编译 2 关卡视口中选中这个蓝图类Door bp的实例 世界大纲视图下的细节面板中 默认下出现公有变量名称NewVar 1 用吸管吸取关
  • springboot+poi开发excel导出 加载Excel模板导出 Excel导出详解

    提到Excel导出功能 可能很多人都使用springmvc框架做过 笔者今天要给大家分享的是基于springBoot开发Excel复杂模板导出功能 所谓复杂模板指在模板里的特定表头里有不同的单元格合并以及背景色 字体颜色的填充 文本内容的对
  • linux下libpcap抓包分析

    linux下libpcap抓包分析 一 首先下载libpcap包http www tcpdump org latest release 然后安装 安装完成后进入安装根目录的tests文件夹 编译运行findalldevstest c 编译时
  • 实现mnist手写数字识别(第一周)

    本文为 365天深度学习训练营 中的学习记录博客 参考文章 365天深度学习训练营 第P1周 实现mnist手写数字识别 Pytorch实战 第P1周 实现mnist手写数字识别 qq com 原作者 K同学啊 接辅导 项目定制 我的环境
  • 使用codestriker搭建代码评审平台

    codestriker是用perl语言开发的 可以使用apache cgi进行访问的代码评审web站点 搭建过程如下 1 yum install perl 2 yum install highlight 3 配置codestriker co
  • k8s部署minio

    安装krew插件 官网地址 https krew sigs k8s io docs user guide setup install set x cd mktemp d OS uname tr upper lower
  • 图论算法<三>:判断有向图中是否有存在循环 ,以及环的个数和各个环中的元素

    1 目的 判断有向图中是否有存在循环 以及环的个数和各个环中的元素 2 示例效果 2 1 原始数据 路线起终点整理如下 共计12个顶点 19条边 起点 终点 1 最后的1代表起点终点是连通的 起点 终点 1 2 4 1 起点 终点 1 9