先看一个有意思的问题, 我们定义一个二维数组表示迷宫。

它表示一个迷宫, 其中的1表示墙壁,0表示可以走的路, 只能横着走或竖着走,不能斜着走,

我们要编程序找出从左上角到右下角的路线。其实这个问题可以用深度优先搜索的方法搞定的了。

这个算法中涉及到了几个知识:

1. 一个是栈,每走过一个点会把这个点会把这个点压入栈中。

2. 用一个新的数据结构保存走迷宫的路线,每个走过的点都有一个前趋(Predecessor)的点,表示是从哪儿走到

当前点的,比如predecessor[4][4]是座标为(3, 4)的点,就表示从(3, 4)走到了(4, 4),一开始predecessor的各元素

初始化为无效座标(-1, -1)。在迷宫中探索路线的同时就把路线保存在predecessor数组中。

3. 已经走过的点在maze数组中记为2防止重复走。 算法的伪代码如下 :

分为几个文件来分开实现 :

/* main.c */

/***********************************************************/
/* Copyright (C) SA14226214, USTC, 2014-2015 */
/* */
/* FILE NAME : main.c */
/* PRINCIPAL AUTHOR : GaoZhipeng */
/* SUBSYSTEM NAME : MakFile */
/* MODULE NAME : makefile */
/* LANGUAGE : C */
/* TARGET ENVIRONMENT : ANY */
/* DATE OF FIRST RELEASE : 2015/04/05 */
/* DESCRIPTION : This is a makefile program */
/***********************************************************/ /*
*Revision log:
*
*Ceated by GaoZhipeng, 2015/04/05
*
*/ #include<stdio.h>
#include"main.h"
#include"stack.h"
#include"maze.h" struct point predecessor[MAX_ROW][MAX_COL] = {
{{-,-}, {-,-}, {-,-}, {-,-}, {-,-}},
{{-,-}, {-,-}, {-,-}, {-,-}, {-,-}},
{{-,-}, {-,-}, {-,-}, {-,-}, {-,-}},
{{-,-}, {-,-}, {-,-}, {-,-}, {-,-}},
{{-,-}, {-,-}, {-,-}, {-,-}, {-,-}},
}; void visit(int row, int col, struct point pre)
{
struct point visit_point = {row, col};
maze[row][col] = ;
predecessor[row][col] = pre;
push(visit_point);
} int main(void)
{
struct point p = { , };
maze[p.row][p.col] = ;
push(p);
while (!is_empty()) {
p = pop();
if (p.row == MAX_ROW - /* goal */
&& p.col == MAX_COL - )
break;
if (p.col+ < MAX_COL
/* right */
&& maze[p.row][p.col+] == )
visit(p.row, p.col+, p);
if (p.row+ < MAX_ROW
/* down */
&& maze[p.row+][p.col] == )
visit(p.row+, p.col, p);
if (p.col- >=
/* left */
&& maze[p.row][p.col-] == )
visit(p.row, p.col-, p);
if (p.row- >=
/* up */
&& maze[p.row-][p.col] == )
visit(p.row-, p.col, p);
print_maze();
}
if (p.row == MAX_ROW - && p.col == MAX_COL - ) {
printf("(%d, %d)\n", p.row, p.col);
while (predecessor[p.row][p.col].row != -) {
p = predecessor[p.row][p.col];
printf("(%d, %d)\n", p.row, p.col);
}
} else
printf("No path!\n");
return ;
}

/* main.h */

/***********************************************************/
/* Copyright (C) SA14226214, USTC, 2014-2015 */
/* */
/* FILE NAME : main.h */
/* PRINCIPAL AUTHOR : GaoZhipeng */
/* SUBSYSTEM NAME : MakFile */
/* MODULE NAME : makefile */
/* LANGUAGE : C */
/* TARGET ENVIRONMENT : ANY */
/* DATE OF FIRST RELEASE : 2015/04/05 */
/* DESCRIPTION : This is a makefile program */
/***********************************************************/ /*
*Revision log:
*
*Ceated by GaoZhipeng, 2015/04/05
*
*/ #ifndef MAIN_H
#define MAIN_H typedef struct point{
int row, col;
} item_t; #define MAX_ROW 5
#define MAX_COL 5 #endif

/* maze.c */

/***********************************************************/
/* Copyright (C) SA14226214, USTC, 2014-2015 */
/* */
/* FILE NAME : maze.c */
/* PRINCIPAL AUTHOR : GaoZhipeng */
/* SUBSYSTEM NAME : MakFile */
/* MODULE NAME : makefile */
/* LANGUAGE : C */
/* TARGET ENVIRONMENT : ANY */
/* DATE OF FIRST RELEASE : 2015/04/05 */
/* DESCRIPTION : This is a makefile program */
/***********************************************************/ /*
*Revision log:
*
*Ceated by GaoZhipeng, 2015/04/05
*
*/ #include<stdio.h>
#include"maze.h" int maze[MAX_ROW][MAX_COL] = {
, , , , ,
, , , , ,
, , , , ,
, , , , ,
, , , , ,
}; void print_maze(void)
{
int i, j;
for(i=; i<MAX_ROW; i++)
{
for(j=; j<MAX_COL; j++)
{
printf("%d ", maze[i][j]);
}
putchar('\n');
} printf("*********\n");
}

/* maze.h */

/***********************************************************/
/* Copyright (C) SA14226214, USTC, 2014-2015 */
/* */
/* FILE NAME : maze.h */
/* PRINCIPAL AUTHOR : GaoZhipeng */
/* SUBSYSTEM NAME : MakFile */
/* MODULE NAME : makefile */
/* LANGUAGE : C */
/* TARGET ENVIRONMENT : ANY */
/* DATE OF FIRST RELEASE : 2015/04/05 */
/* DESCRIPTION : This is a makefile program */
/***********************************************************/ /*
*Revision log:
*
*Ceated by GaoZhipeng, 2015/04/05
*
*/ #ifndef MAZE_H
#define MAZE_H #include"main.h" extern int maze[MAX_ROW][MAX_COL];
void print_maze(void); #endif

/* stack.c */

/***********************************************************/
/* Copyright (C) SA14226214, USTC, 2014-2015 */
/* */
/* FILE NAME : stack.c */
/* PRINCIPAL AUTHOR : GaoZhipeng */
/* SUBSYSTEM NAME : MakFile */
/* MODULE NAME : makefile */
/* LANGUAGE : C */
/* TARGET ENVIRONMENT : ANY */
/* DATE OF FIRST RELEASE : 2015/04/05 */
/* DESCRIPTION : This is a makefile program */
/***********************************************************/ /*
*Revision log:
*
*Ceated by GaoZhipeng, 2015/04/05
*
*/ #include"stack.h" static item_t stack[];
static int top = ; void push(item_t p)
{
stack[top++] = p;
} item_t pop(void)
{
return stack[--top];
} int is_empty(void)
{
return top == ;
}

/* stack.h */

/***********************************************************/
/* Copyright (C) SA14226214, USTC, 2014-2015 */
/* */
/* FILE NAME : stack.h */
/* PRINCIPAL AUTHOR : GaoZhipeng */
/* SUBSYSTEM NAME : MakFile */
/* MODULE NAME : makefile */
/* LANGUAGE : C */
/* TARGET ENVIRONMENT : ANY */
/* DATE OF FIRST RELEASE : 2015/04/05 */
/* DESCRIPTION : This is a makefile program */
/***********************************************************/ /*
*Revision log:
*
*Ceated by GaoZhipeng, 2015/04/05
*
*/ #ifndef STACK_H
#define STACK_H #include "main.h" extern void push(item_t);
extern item_t pop(void);
extern int is_empty(void); #endif

然后我们要做的就是在Linux下对上述程序进行编译和运行。

对于程序的编译命令, 最原始的编译指令可以这样写:  $ gcc main.c stack.c maze.c -o main

不过还有一个更加简便的操作, 那就是写一个Makefile文件和源代码放在同一个目录下面。下面是一个写好的Makefile

# This is an simple makefile

main : main.o stack.o maze.o
gcc main.o stack.o maze.o -o main main.o : main.c main.h stack.h maze.h
gcc -c main.c -o main.o stack.o : stack.c stack.h main.h
gcc -c stack.c -o stack.o maze.o : maze.c maze.h main.h
gcc -c maze.c -o maze.o clean :
@echo "cleanning the project"
-rm main *.o
@echo "cleanning completed" .PHONY : clean

make 命令会自动读取当前目录下的Makefile 文件, 完成相应的编译步骤。Makefile由一组规则(Rule)组成,每条规则的格式是:

例如 :  main: main.o stack.o maze.o
      gcc main.o stack.o maze.o -o main

main 是这条规则的目标(Target),main.o、stack.o 和maze.o是这条规则的条件(Prerequisite)。目标和条件之间的关系是:欲更新目标,

必须首先更新它的所有条件;所有条件中只要有一个条件被更新了,目标也必须随之被更新。所谓“更新”就是执行一遍规则中的命令列表,

命令列表中的每条命令必须以一个Tab开头,注意不能是空格,Makefile的格式不像C语言的缩进那么随意,对于Makefile中的每个以Tab

开头的命令,make 会创建一个Shell进程去执行它。

Makefile中有一些常用的特殊变量 :

  • $@ ,表示规则中的目标。
  • $< ,表示规则中的第一个条件。
  • $?,表示规则中所有比目标新的条件,组成一个列表,以空格分隔。
  • $^ ,表示规则中的所有条件,组成一个列表,以空格分隔。

那么我们可以把Makefile进一步化简, 写成如下的形式:

#this is a makefile

all : main

main : main.o stack.o maze.o
gcc $^ -o $@ main.o : main.h stack.h maze.h
stack.o : stack.h main.h
maze.o : maze.h main.h clean :
-rm main *.o .PHONY: clean

把命令 gcc main.o stack.o maze.o -o main 改写成 gcc $^ -o $@。

这样即使以后又往条件里添加了新的目标文件,编译命令也不需要修改,减少了出错的可能。

迷宫问题&MakeFile的更多相关文章

  1. 十天学Linux内核之第十天---总结篇(kconfig和Makefile & 讲不出再见)

    原文:十天学Linux内核之第十天---总结篇(kconfig和Makefile & 讲不出再见) 非常开心能够和大家一起分享这些,让我受益匪浅,感激之情也溢于言表,,code monkey的 ...

  2. 24小时学通Linux内核总结篇(kconfig和Makefile & 讲不出再见)

    非常开心能够和大家一起分享这些,让我受益匪浅,感激之情也溢于言表,,code monkey的话少,没办法煽情了,,,,,,,冬天的风,吹得伤怀,倒叙往事,褪成空白~学校的人越来越少了,就像那年我们小年 ...

  3. Linux内核(1) - Kernel地图:Kconfig与Makefile

    Makefile不是Make Love 从前在学校,混了四年,没有学到任何东西,每天就是逃课,上网,玩游戏,睡觉.毕业的时候,人家跟我说Makefile我完全不知,但是一说Make Love我就来劲了 ...

  4. 《Linux内核修炼之道》精华分享与讨论(5)——Kernel地图:Kconfig与Makefile

    转自:http://blog.csdn.net/fudan_abc/article/details/5340408 Makefile不是Make Love 从前在学校,混了四年,没有学到任何东西,每天 ...

  5. 说说Makefile那些事儿

    说说Makefile那些事儿 |扬说|透过现象看本质 工作至今,一直对Makefile半知半解.突然某天幡然醒悟,觉得此举极为不妥,只得洗心革面从头学来,以前许多不明觉厉之处顿时茅塞顿开,想想好记性不 ...

  6. C语言动态走迷宫

    曾经用C语言做过的动态走迷宫程序,先分享代码如下: 代码如下: //头文件 #include<stdio.h> #include<windows.h>//Sleep(500)函 ...

  7. 编写一个通用的Makefile文件

    1.1在这之前,我们需要了解程序的编译过程 a.预处理:检查语法错误,展开宏,包含头文件等 b.编译:*.c-->*.S c.汇编:*.S-->*.o d.链接:.o +库文件=*.exe ...

  8. 编写简单的Makefile文件

    makefile中的编写内容如下: www:hello.c x.h gcc hello.c -o hello clean: rm hello www:hello.c  x.h 表示生成www这个文件需 ...

  9. 简单编写Makefile

    相信很多朋友都有过这样的经历,看着开源项目中好几页的makefile文件,不知所云.在日常学习和工作中,也有意无意的去回避makefile,能改就不写,能用ide就用ide.其实makefile并没有 ...

随机推荐

  1. 给Visual Studio更换皮肤和背景图

    1.先安装更换皮肤的插件 VS菜单栏里面找到:工具>扩展和更新>联机>搜索: Theme Editor 下载并安装: 安装后先不着急重启VS 然后,安装可更改背景图片的插件:工具&g ...

  2. 字符串在forEach方法里面可以使用include函数

    今天在写项目的时候,发现了一个问题,使用forEach函数,arr数组里面的字符串可以调用include方法,我查阅了很多地方,string里面没有这个方法. 但是在forEach函数里面确实可以这样 ...

  3. VC中包含的头文件名不区分大小写

    VC中包含的头文件名,不区分大小写如 #include "my.h" = #include "MY.H".

  4. Python3+Selenium3+webdriver学习笔记12(js操作应用:滚动条 日历 内嵌div)

    #!/usr/bin/env python# -*- coding:utf-8 -*-'''Selenium3+webdriver学习笔记12(js操作应用:滚动条 日历 内嵌div)'''from ...

  5. github上fork原项目,如何将本地仓库代码更新到最新版本?

    场景: 在github上fork原项目,项目组成员发起pull request提交了代码,这时自己在本地仓库该如何更新到最新代码? 操作方法如下: 方法一.从github上进行操作然后更新 登录自己的 ...

  6. 轮播插件unslider.min.js使用demo

    有两种应用方式: 1.轮播图片作为<img>标签使用 HTML代码: <html> <head> <meta charset="utf-8" ...

  7. LibreOJ #514. 「LibreOJ β Round #2」模拟只会猜题意

    内存限制:256 MiB 时间限制:1000 ms 标准输入输出 题目类型:传统 评测方式:文本比较 题目描述 给定一个长度为 nnn 的序列 AAA . 定义 f(l,r)=∑i=lrAif(l,r ...

  8. Codeforces Round #411 div2

    A. Fake NP 题意:询问一个区间[L,R]出现次数最多的正整数因子(>1). 一个区间内一个因子P出现次数大概为[R/P]-[(L-1)/P],约等于(R-L+1)/P,P取2时最优.注 ...

  9. Android(java)学习笔记111:Java异常分类

    Java异常可分为3种: (1)编译时异常:Java.lang.Exception (2)运行期异常:Java.lang.RuntimeException (3)错误:Java.lang.Error

  10. kubernetes监控-prometheus(十六)

    监控方案 cAdvisor+Heapster+InfluxDB+Grafana Y 简单 容器监控 cAdvisor/exporter+Prometheus+Grafana Y 扩展性好 容器,应用, ...