一直以来觉得图论相关方面的算法是用来拔高的,是哪些搞ACM的活,对于我等无人之辈了解了解就行,但是后来在华为的编程大赛以及校招机试中均遇到涉及图论相关算法的题目(分值还蛮高的),结果是一筹莫展啊!下来看别人提交的代码好像也不长,于是在网上找了些相关代码看了看,自己试着开始写写,顺便总结总结,发现也没有想象中那么复杂。
下面是华为编程大赛中的一道题目:
分析题目,题目中给出的已知条件有:地图、起始点B(笨笨熊现在的房子)、终点H(笨笨熊新的豪宅),题目要求给出笨笨熊从现在的房子是否可到达其新到的房子,也就是说在地图中从点B是否有一条路径可达点H。这是典型的深度优先搜索DFS/BFS题。对于图我们可以抽象理解为:图的顶点表示某种状态(白、灰、黑),图的边表示顶点与顶点之间的某种规则。对应于此题,我们可以用一个二维数组存储地图,地图中的每个顶点/路径即:数组的每个元素代表一种状态,表示当前路径是否可行,而笨笨熊可以向上下左右四个方向行走,则代表图的边。我们可以用一个二维数组rule[4][2] = { {-1, 0}, {0, -1}, {1, 0}, {0, 1} }来表示这种行走规则/图的边。
以下是DFS的代码、注释及运行结果图:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
typedef struct
{
int x;
int y;
}MapPos;
enum State
{
BLOCK = 0, // 此路径不可通过
PASS, // 此路径可通过
VISITED // 此路径已走过
};
const int MAX_LEN = 10; // 地图行列数最大值
State Map[MAX_LEN][MAX_LEN] = {BLOCK}; // 地图
// int PathLen[MAX_LEN][MAX_LEN] = {-1}; // 存储起始点到当前点的路径长度 -1 --- 不可达
int m = 0, n = 0; // 当前输入地图的行列数
int IsAccess = 0; // 是否可达(0/1 ----- 不可达/可达)
MapPos fromPos, endPos; // 起始点,目的点
void input( int m, int n );
void Dfs( int x, int y );
int main()
{
char c;
while( true )
{
scanf("%d", &m); // 获取地图大小(行数)
scanf("%d", &n); // 获取地图大小(列数),scanf函数会跳过输入流缓冲区中的数据之前的空白符(' '、'\t'、'\n'),所以获取到的是4而不是'\n'
fflush( stdin ); // 刷新输入缓冲区(此时stdin缓冲区中第一个字符为'\n',而gets/getline函数遇到换行符会结束)
input( m, n ); // 根据输入获取地图
Dfs( fromPos.x, fromPos.y );
printf( "%s\n", 1==IsAccess?"Y":"N" );
printf( "输入q结束 or 其他字符重新开始\n" );
fflush( stdin );
scanf( "%c", &c );
if( 'q' == c )
break;
// 复位
memset( Map, BLOCK, MAX_LEN * MAX_LEN *sizeof(int) );
IsAccess = 0;
}
system( "pause" );
return 0;
}
/*************************************