问题 : 算法7-6:图的遍历——广度优先搜索

时间限制: 1Sec 内存限制: 32MB

题目描述

广度优先搜索遍历类似于树的按层次遍历的过程。其过程为:假设从图中的某顶点v出发,在访问了v之后依次访问v的各个未曾被访问过的邻接点,然后分别从这些邻接点出发依次访问它们的邻接点,并使“先被访问的顶点的邻接点”先于“后被访问的顶点的邻接点”被访问,直至图中所有已被访问的顶点的邻接点都被访问到。若此时图中尚有顶点未被访问,则另选图中一个未曾被访问的顶点作为起始点。重复上述过程,直至图中所有顶点都被访问到为止。
其算法可以描述如下:
在本题中,读入一个无向图的邻接矩阵(即数组表示),建立无向图并按照以上描述中的算法遍历所有顶点,输出遍历顶点的顺序。

输入

输入的第一行包含一个正整数n,表示图***有n个顶点。其中n不超过50。
以后的n行中每行有n个用空格隔开的整数0或1,对于第i行的第j个0或1,1表示第i个顶点和第j个顶点有直接连接,0表示没有直接连接。当i和j相等的时候,保证对应的整数为0。
输入保证邻接矩阵为对称矩阵,即输入的图一定是无向图。

输出

只有一行,包含n个整数,表示按照题目描述中的广度优先遍历算法遍历整个图的访问顶点顺序。每个整数后输出一个空格,并请注意行尾输出换行。

样例输入

4
0 0 0 1
0 0 1 1
0 1 0 1
1 1 1 0

样例输出

0 3 1 2 

提示

在本题中,需要熟练掌握图的邻接矩阵存储方式。在建立完成无向图之后,需要严格按照题目描述的遍历顺序对图进行遍历。另外,算法中描述的FirstAdjVex函数和NextAdjVex函数,需要认真的自行探索并完成。在本题中需要使用队列结构,需要对队列的概念进行复习。
通过这道题目,应该能够对图的广度优先搜索建立更加直观和清晰的概念。

解题思路

在本题中,我主要借用了啊哈算法中队列的进列和出列,参考着啊哈算法中的广度优先搜索写的。广度搜索主要还是层层递进,在这里主要运用了队列,这道题是从0开始搜索的,先找到与0相连的点,然后向外 层层递进,把走过的点插入到队列中去。具体见代码:
#include <stdio.h>
#include <string.h>
int main()
{
    int n, head, tail;
    int arr[2500], map[55][55], vis[55];
    while (~scanf("%d", &n))
    {
	head = tail = 0;
	memset(vis, 0, sizeof(vis));
	for (int i = 0; i < n; i++)
	{
       	    for (int j = 0; j < n; j++)
	        scanf("%d", &map[i][j]);
	}
	vis[0] = 1;
	arr[tail++] = 0;
	printf("0 ");
	while (head < tail)
	{
	    for (int i = 0; i < n; i++)
       	    {
		if (map[arr[head]][i] && !vis[i])
		{
		    vis[i] = 1;
		    arr[tail++] = i;
		    printf("%d ", i);
		}
	    }
	    head++;
	}
	puts("");
    }
    return 0;
}