c语言数据结构--------拓扑排序和逆拓扑排序(Kahn算法和DFS算法实现)

news2025/4/7 17:11:00
#include <stdio.h>
#include <string.h>
#include <stdbool.h>
#include <stdlib.h>


//使用卡恩算法(Kahn)和深度优先算法(DFS)实现

//拓扑排序和逆拓扑排序

//拓扑排序和逆拓扑排序顶点顺序相反

//图,邻接矩阵存储
#define MaxVertexNum 100  //最大顶点数

typedef struct {
    int vex[MaxVertexNum]; //顶点表
    int edge[MaxVertexNum][MaxVertexNum]; //边表
    int vernum, arcnum; //记录当前图的顶点数量和边数
} MGraph;

int vexIndex(MGraph G, int x);

void visit(int i);

//初始化图
MGraph InitMgraph() {
    MGraph graph;
    memset(graph.edge, 0, sizeof(graph.edge));
    graph.arcnum = 0;
    graph.vernum = 0;
    return graph;
}



//带头节点
//链队列节点
typedef struct Linknode {
    int data;
    struct Linknode *next;
} Linknode;

//链队列
typedef struct {
    //队头指针
    Linknode *front;
    //队尾指针
    Linknode *rear;
} LinkQueue;

//初始化队列
void InitQueue(LinkQueue *Q) {
    //创建头结点
    Linknode *temp = (Linknode *) malloc(sizeof(Linknode));
    Q->front = temp;
    Q->rear = temp;
}

//入队
void EnQueue(LinkQueue *Q, int data) {
    //创造节点
    Linknode *temp = (Linknode *) malloc(sizeof(Linknode));
    //赋值
    temp->data = data;
    temp->next = NULL;
    //连接插入节点
    Q->rear->next = temp;
    //队尾指针更换
    Q->rear = temp;
}

//出队
bool DeQueue(LinkQueue *Q, int *e) {
    //队列为空
    if (Q->rear == Q->front) {
        return false;
    }
    //要出队的节点
    Linknode *temp = Q->front->next;
    *e = temp->data;
    //队头指针更换
    Q->front->next = temp->next;
    free(temp);
    //最后一个节点出队
    if (temp == Q->rear) {
        //队尾指针更换
        Q->rear = Q->front;
    }
    return true;
}

/
//借助队列实现拓扑排序
//卡恩算法
bool TopologicalSort(MGraph G) {

    //初始化队列
    LinkQueue *linkQueue = (Linknode *) malloc(sizeof(linkQueue));
    InitQueue(linkQueue);


    //当前顶点的入度
    int indegree[G.vernum];
    //记录拓扑排序
    int print[G.vernum];

    memset(indegree, 0, sizeof(indegree));
    //初始化入度数组
    for (int i = 0; i < G.vernum; ++i) {
        for (int j = 0; j < G.vernum; j++) {
            if (G.edge[j][i] == 1) {
                indegree[i]++;
            }
        }
    }

    //初始化print数组
    memset(print, -1, sizeof(print));

    //度为0的顶点索引入队列
    for (int i = 0; i < G.vernum; i++) {
        if (indegree[i] == 0) {
            EnQueue(linkQueue, i);
            indegree[i] = -1;
        }
    }
    //记录顶点个数
    int count = 0;
    int *e = malloc(sizeof(int));
    while (linkQueue->rear != linkQueue->front) {
        DeQueue(linkQueue, e);
        print[count] = G.vex[*e];

        count++;

        for (int i = 0; i < G.vernum; ++i) {
            if (G.edge[*e][i] == 1) {
                indegree[i]--;
            }

            if (indegree[i] == 0) {
                EnQueue(linkQueue, i);
                indegree[i] = -1;
            }

        }
    }

    if (count < G.vernum)
        return false;
    else {
        for (int i = 0; i < G.vernum; i++) {
            printf("%d  ", print[i]);
        }
        return true;
    }

}

int time;
int finalTime[100];
int visited[100];

//DFS算法实现拓扑排序
//v为入度为0的顶点
void DFSTopologicalSort(MGraph G, int i) {
    //对i做已访问标记
    visited[vexIndex(G, i)] = 1;
    //找到i的所有邻接点
    for (int j = 0; j < G.vernum; j++) {
        if (visited[j] == 0 && G.edge[vexIndex(G, i)][j] == 1) {
            DFSTopologicalSort(G, G.vex[j]);
        }
    }
    (time)++;
    finalTime[vexIndex(G, i)] = time;

}

//使用后visited会被重新赋值,需重置
//建议用一个变量临时保存原有图
void DFSTraverseTopologicalSort(MGraph G, int i) {
    time = 0;
    for (int j = 0; j < G.vernum; j++) {
        visited[j] = 0;
    }
    DFSTopologicalSort(G, i);
    for (int j = 0; j < G.vernum; j++) {
        if (visited[j] == 0) {
            DFSTopologicalSort(G, G.vex[j]);
        }
    }

    for (int j = 0; j < G.vernum; j++) {
        printf("%d  ", finalTime[j]);
    }
}


//有向图深度优先搜索
void niDFSTopologicalSort(MGraph G, int i) {

    //对i做已访问标记
    visited[vexIndex(G, i)] = 1;
    //找到i的所有邻接点
    for (int j = 0; j < G.vernum; j++) {
        if (visited[j] == 0 && G.edge[vexIndex(G, i)][j] == 1) {
            niDFSTopologicalSort(G, G.vex[j]);
        }
    }
    printf("%d  ", i);
}

//使用后visited会被重新赋值,需重置
//建议用一个变量临时保存原有图
//连通图和非连通图的深度优先搜索(改进)
void niDFSTraverseTopologicalSort(MGraph G, int i) {
    for (int j = 0; j < G.vernum; j++) {
        visited[j] = 0;
    }

    niDFSTopologicalSort(G, i);
    for (int j = 0; j < G.vernum; j++) {
        if (visited[j] == 0) {
            niDFSTopologicalSort(G, G.vex[j]);
        }
    }
}


int main(void) {
    //有向图
    //初始化图
    MGraph graph = InitMgraph();
    //图添加元素
    //顶点集添加1,2,3,4,5
    for (int i = 0; i < 5; i++) {
        graph.vex[i] = i + 1;
        graph.vernum++;
    }
    //边集加边<1,2>,<1,4>,<2,3>,<2,4>,<3,5>,<4,3>,<4,5>
    graph.edge[0][1] = 1;
    graph.edge[0][3] = 1;
    graph.edge[1][2] = 1;
    graph.edge[1][3] = 1;
    graph.edge[2][4] = 1;
    graph.edge[3][2] = 1;
    graph.edge[3][4] = 1;
    graph.arcnum = 7;
    printf("拓扑排序序列:\n");
    TopologicalSort(graph);
    printf("\n");
    printf("tineFinal数组:\n");
    DFSTraverseTopologicalSort(graph, 1);
    printf("\n");
    printf("逆拓扑排序序列:\n");
    niDFSTraverseTopologicalSort(graph, 5);
}

//返回x在顶点表的索引
int vexIndex(MGraph G, int x) {
    int index = -1;
    for (int i = 0; i < G.vernum; i++) {
        if (G.vex[i] == x) {
            index = i;
            break;
        }
    }
    return index;
}

测试用例

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2328273.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

谷粒微服务高级篇学习笔记整理---nginx搭建正反向代理

正向与反向代理 **正向代理:**客户端向代理服务器发请求并指定目标服务器,代理向目标转交请求并将获得的内容转给客户端。 反向代理:用户直接访问反向代理服务器就可以获得目标服务器的资源。反向代理服务器统一了访问入口。 给首页配置反向代理 修改windows的hosts文件配…

2.pycharm保姆级安装教程

一、pycharm安装 1.官网上下载好好软&#xff0c;双击打开 2.下一步 3.修改路径地址 (默认也可以) 4.打勾 5.安装 不用重启电脑 二、添加解释器 1.双击软件&#xff0c;打开 2.projects – new project 3.指定项目名字&#xff0c;项目保存地址&#xff0c;解释器 4.右击 – …

【SQL】取消sql某一列的唯一值key值的方法

在插入数据到sql时&#xff0c;遇到了这个问题&#xff1a; Duplicate entry ‘XXX’ for key 起因是&#xff1a; 我之前设计表的时候&#xff0c;手动给product_title 这个列加了一个key&#xff0c; key 是这个字段的唯一键约束&#xff0c;就不能重复在这一列存入重复的数…

数据库--SQL

SQL&#xff1a;Structured Query Language&#xff0c;结构化查询语言 SQL是用于管理关系型数据库并对其中的数据进行一系列操作&#xff08;包括数据插入、查询、修改删除&#xff09;的一种语言 分类&#xff1a;数据定义语言DDL、数据操纵语言DML、数据控制语言DCL、事务处…

SQL语句(一)—— DDL

目录 一、SQL 基础知识 &#xff08;一&#xff09;SQL 通用语法 &#xff08;二&#xff09;SQL 分类 二、DDL —— 数据库操作 1、查询所有数据库 2、查询当前数据库 3、创建数据库 4、删除数据库 5、切换数据库 三、DDL —— 表操作 &#xff08;一&#xff09;查…

Husky目标跟踪

1.0设备清单 幻影峡谷、适配器 摄像头及数据线、显卡欺骗器 外接屏幕、键盘鼠标 Husky底盘、便携显示屏、键盘鼠标 移动电源 1.1连线 插排——移动电源幻影峡谷——适配器——插排摄像头——幻影峡谷&#xff08;摄像头固定在机械臂前方的底盘上&#xff09;键盘鼠标显示器…

Python----机器学习(线性回归:自求导的方法实现)

一、线性回归方程 目标&#xff1a; 线性回归的目标是找到最佳的系数来使模型与观察到的数据尽可能拟合。 应用&#xff1a; 预测&#xff1a;给定自变量的值&#xff0c;预测因变量的值。 回归分析&#xff1a;确定自变量对因变量的影响程度 线性回归是统计学和机器学习中最简…

Springcache+xxljob实现定时刷新缓存

目录 SpringCache详解 SpringCache概述 核心原理 接口抽象与多态 AOP动态代理 核心注解以及使用 公共属性 cacheNames KeyGenerator&#xff1a;key生成器 key condition&#xff1a;缓存的条件&#xff0c;对入参进行判断 注解 xxl-job详解 SpringcacheRedis实现…

vue2拖拉拽做个模拟公式工具

1. 成图 2. 介绍 就是简单拖拉拽来做个规则运算器&#xff0c;具体运算规则、校验规则自己加。 3. 代码 HTML代码 <template><div class"red-cont"><div class"red-top"><divclass"red-top-left"><div class&quo…

Windows查重工具,强烈推荐大家收藏!

我大家在用电脑的时候&#xff0c;是不是发现用得越久&#xff0c;电脑里的软件和文件就越多&#xff1f; 今天我给大家带来的这两款重复文件查找神器&#xff0c;简直就是电脑里的“清洁小能手”&#xff0c;能帮你把那些重复的文件和文件夹找出来。 Easy DupLicate Finder 重…

使用python完成手写数字识别

入门图像识别的第一个案例,看到好多小伙伴分享,也把自己当初的思路捋捋,写成一篇博客,作为记录和分享,也欢迎各位交流讨论。 实现思路 数据集:MNIST(包含60,000个训练样本和10,000个测试样本) 深度学习框架:Keras(基于TensorFlow) 模型架构:卷积神经网络(CNN) 实…

OpenLayers:如何控制Overlay的层级?

我最近在使用Overlay的时候遇到了一个问题&#xff0c;我向地图中添加了两种不同的Overlay&#xff08;下图中的蓝色标牌和粉色标牌&#xff09;&#xff0c;我希望粉色标牌可以显示在最上层&#xff0c;可偏偏蓝色标牌却将其遮挡住了。于是我对Overlay的层级开始起了兴趣&…

《Golang高性能网络编程:构建低延迟服务器应用》

在本文中&#xff0c;我们将深入探讨Golang高性能网络编程&#xff0c;帮助您构建低延迟服务器应用。我们将介绍Golang的网络编程特性、优化技巧和实际案例&#xff0c;让您更好地理解和应用Golang在网络编程领域的优势。 高性能网络编程简介 什么是Golang高性能网络编程 高性能…

数据结构C语言练习(设计循环队列)

一、循环队列简介 循环队列是一种线性数据结构&#xff0c;基于 FIFO&#xff08;先进先出&#xff09;原则&#xff0c;将队尾连接到队首形成循环。其核心优势是能复用队列之前用过的空间&#xff0c;避免普通队列 “假溢出” 问题。实现时&#xff0c;通常申请 k1 大小的数组…

vscode代码片段的设置与使用

在 Visual Studio Code (VS Code) 中&#xff0c;可以通过自定义**代码片段&#xff08;Snippets&#xff09;**快速插入常用代码模板。以下是详细设置步骤&#xff1a; 步骤 1&#xff1a;打开代码片段设置 按下快捷键 Ctrl Shift P&#xff08;Windows/Linux&#xff09;或…

uniapp -- 列表垂直方向拖拽drag组件

背景 需要在小程序中实现拖拽排序功能,所以就用到了m-drag拖拽组件,在开发的过程中,发现该组件在特殊的场景下会有些问题,并对其进行了拓展。 效果 组件代码 <template><!-- 创建一个垂直滚动视图,类名为m-drag --><scroll

一款非常小的软件,操作起来非常丝滑!

今天我想给大家分享一款超级实用的小软件&#xff0c;它是一款电脑上用的倒计时和关机助手。 关机助手 帮你自动关机 这款关机助手特别小巧&#xff0c;完全不需要安装&#xff0c;文件大小才60KB&#xff0c;比一个小小的文件还小。你只需要把它下载下来&#xff0c;双击打开…

FrameWork基础案例解析(四)

文章目录 单独拉取framework开机与开机动画横屏Android.mk语法单独编译SDKmake 忽略warning单独修改和编译Camera2单独编译Launcher3Android Studio 导入、修改、编译Settings导入 Android Studio 导入、修改、编译Launcher3android 开机默认进入指定Launcher植入自己的apk到系…

通过 C# 提取PDF文档中的图片

当 PDF 文件中包含有价值的图片&#xff0c;如艺术画作、设计素材、报告图表等&#xff0c;提取图片可以将这些图像资源进行单独保存&#xff0c;方便后续在不同的项目中使用&#xff0c;避免每次都要从 PDF 中查找。本文将介绍如何使用C#通过代码从PDF文档中提取图片&#xff…

国标GB28181视频监控平台EasyCVR保驾护航休闲娱乐“九小场所”安全运营

凭借降低人力资源、节约物资成本的优势&#xff0c;在多个场景得到广泛应用。如今&#xff0c;棋牌室、洗浴中心、酒店这类人员频繁流动和密集的场所&#xff0c;已成为安全管理的重点。​ 尽管部分棋牌室已安装了监控设备&#xff0c;但是设备功能单一&#xff0c;只能实现一…