数据结构实验任务七:基于广度优先搜索的六度空间理论验证

本文主要是介绍数据结构实验任务七:基于广度优先搜索的六度空间理论验证,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

问题描述

“六度空间”理论又称作“六度分隔(Six Degrees of Separation)”理论。这个理论 可以通俗地阐述为:“你和任何一个陌生人之间所间隔的人不会超过六个,也就是 说,最多通过五个人你就能够认识任何一个陌生人。”假如给你一个社交网络图, 请你对每个节点计算符合“六度空间”理论的结点占结点总数的百分比。 输入要求 多组数据,每组数据 m+1 行。第一行有两个数字 n 和 m,代表有 n 个人和 m 组朋友关系。n 个人的编号为 1 到 n。第二行到第 m+1 行每行包括两个数字 a 和 b,代表这两个人互相认识。当 n 和 m 都等于 0 时,输入结束。 输出要求 每组数据输出 n 行,对每个结点输出与该结点距离不超过 6 的结点数占结点 总数的百分比,精确到小数点后 2 位。每个结节点输出一行,格式为“结点编号:(空 格)百分比%”

运行结果:

代码实现:

#include <stdio.h>
#include <stdlib.h>
#define MaxS 20
#define MaxE 5
//结构体部分
typedef struct{						//图结构体定义 char vex[MaxS];					//顶点数组 int vexnum;						//顶点个数 int mat[MaxS][MaxS];			//邻接矩阵 int arcnum;						//边数 
}Graph;
typedef struct{int head,tail;int mat[20];						//队列数组 
}Queue;
//全局变量部分
float result[MaxE][MaxS];					//结果存储函数
int cunt;							//用于全局变量遍历 
int state[MaxS];
//函数声明部分
void Init(Graph *a);
int Read(Graph *a);
void Cal(Graph *a);
void show();
float BFS(Graph *a,int s);
void InitQ(Queue *q);				//初始化队列 
void push(Queue* q,int n);			//入队 
int pop(Queue *q);					//出队 
//函数定义部分
void InitQ(Queue *q){q->head=0;q->tail=0;
}
void push(Queue* q,int n){q->mat[q->tail] = n;q->tail++;
}
int pop(Queue* q){q->head++;if(q->head>q->tail)return -1;return q->mat[q->head-1];
}
float BFS(Graph *a,int s){int tmp,rs=0;int *len=(int*)malloc(sizeof(int)*a->vexnum+1);			//记录到s的距离 for(int i=0;i<=a->vexnum;i++){state[i] = 0;len[i] = 0;}Queue* q = (Queue*)malloc(sizeof(Queue));InitQ(q);push(q,s);state[s]=1;for(int i = 0;i<a->vexnum;i++){						//总共遍历a->vexnum次tmp = pop(q);if(tmp==-1)continue;for(int j=1;j<=a->vexnum;j++){					//每次扫描vexnum个数  if(a->mat[tmp][j]==1&&state[j]==0){len[j] = len[tmp]+1;state[j] = 1;							//状态变成已访问push(q,j);continue;}}}for(int i=1;i<=a->vexnum;i++){if(len[i]<=6&&len[i]!=0)rs+=1;}for(int i=1;i<=a->vexnum;i++){}free(len);free(q);return (float)(rs+1)/a->vexnum*100;
}
void show(){printf("\n                  =================|    -FZC-    |===============                 \n\n");									printf("FOLLOWING OUTPUT:\n");for(int i=0;i<cunt;i++){printf("[EXP %d ]\n",i+1);for(int j=1;j<MaxS;j++){if(result[i][j]==-1)break;printf("%d: %.2f%%\n",j,result[i][j]);}}
} 
void Init(Graph *a){a->arcnum=0;a->vexnum=0;for(int i=0;i<MaxS;i++){a->vex[i] =0;state[i] = 0;result[cunt][i] = -1;for(int j=0;j<MaxS;j++){a->mat[i][j] = 0;}}}int Read(Graph *a){int n,m,s,e;printf("input n,m:");scanf("%d %d",&n,&m);if(n==0&&m==0)return 1;			//若均为0则返回1 a->vexnum = n;a->arcnum = m;printf("input relationship:\n");for(int i=0;i<m;i++){scanf("%d %d",&s,&e);a->mat[s][e] = 1;a->mat[e][s] = 1; }printf("边输入完成;共%d条\n",a->arcnum);return 0; 
}
void Cal(Graph *a){for(int i=1;i<=a->vexnum;i++){result[cunt][i] = BFS(a,i);}printf("\nSuccess!\n");cunt++; 
}
//主函数部分 
int main(){int flag = 0; Graph* a=(Graph*)malloc(sizeof(Graph));cunt=0;printf("多组数据,每组数据 m+1 行。第一行有两个数字 n 和 m,代表有 n 个人和m 组朋友关系。\nn 个人的编号为 1 到 n。\n第二行到第 m+1 行每行包括两个数字 a和 b,代表这两个人互相认识。\n当 n 和 m 都等于 0 时,输入结束。");while(1){//初始化Init(a);printf("\n                  =================|    -FZC-    |===============                 \n\n");//读取数据flag = Read(a);if(flag==1){show();				//输出结果break; }//处理数据Cal(a); }printf("程序结束!\n"); return 0;
}

这篇关于数据结构实验任务七:基于广度优先搜索的六度空间理论验证的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/466976

相关文章

springboot使用Scheduling实现动态增删启停定时任务教程

《springboot使用Scheduling实现动态增删启停定时任务教程》:本文主要介绍springboot使用Scheduling实现动态增删启停定时任务教程,具有很好的参考价值,希望对大家有... 目录1、配置定时任务需要的线程池2、创建ScheduledFuture的包装类3、注册定时任务,增加、删

Spring Boot 集成 Quartz并使用Cron 表达式实现定时任务

《SpringBoot集成Quartz并使用Cron表达式实现定时任务》本篇文章介绍了如何在SpringBoot中集成Quartz进行定时任务调度,并通过Cron表达式控制任务... 目录前言1. 添加 Quartz 依赖2. 创建 Quartz 任务3. 配置 Quartz 任务调度4. 启动 Sprin

Linux之计划任务和调度命令at/cron详解

《Linux之计划任务和调度命令at/cron详解》:本文主要介绍Linux之计划任务和调度命令at/cron的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux计划任务和调度命令at/cron一、计划任务二、命令{at}介绍三、命令语法及功能 :at

Linux内核参数配置与验证详细指南

《Linux内核参数配置与验证详细指南》在Linux系统运维和性能优化中,内核参数(sysctl)的配置至关重要,本文主要来聊聊如何配置与验证这些Linux内核参数,希望对大家有一定的帮助... 目录1. 引言2. 内核参数的作用3. 如何设置内核参数3.1 临时设置(重启失效)3.2 永久设置(重启仍生效

SpringQuartz定时任务核心组件JobDetail与Trigger配置

《SpringQuartz定时任务核心组件JobDetail与Trigger配置》Spring框架与Quartz调度器的集成提供了强大而灵活的定时任务解决方案,本文主要介绍了SpringQuartz定... 目录引言一、Spring Quartz基础架构1.1 核心组件概述1.2 Spring集成优势二、J

Redis实现延迟任务的三种方法详解

《Redis实现延迟任务的三种方法详解》延迟任务(DelayedTask)是指在未来的某个时间点,执行相应的任务,本文为大家整理了三种常见的实现方法,感兴趣的小伙伴可以参考一下... 目录1.前言2.Redis如何实现延迟任务3.代码实现3.1. 过期键通知事件实现3.2. 使用ZSet实现延迟任务3.3

Linux中的计划任务(crontab)使用方式

《Linux中的计划任务(crontab)使用方式》:本文主要介绍Linux中的计划任务(crontab)使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、前言1、linux的起源与发展2、什么是计划任务(crontab)二、crontab基础1、cro

C#数据结构之字符串(string)详解

《C#数据结构之字符串(string)详解》:本文主要介绍C#数据结构之字符串(string),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录转义字符序列字符串的创建字符串的声明null字符串与空字符串重复单字符字符串的构造字符串的属性和常用方法属性常用方法总结摘

opencv图像处理之指纹验证的实现

《opencv图像处理之指纹验证的实现》本文主要介绍了opencv图像处理之指纹验证的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录一、简介二、具体案例实现1. 图像显示函数2. 指纹验证函数3. 主函数4、运行结果三、总结一、

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用