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

它表示一个迷宫, 其中的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. jQuery:如何给动态生成的元素绑定事件?

    jQuery的html()可以给现在元素附加新的元素,innerHTML也可以,那么,如何给这些新生成的元素绑定事件呢?直接在元素还未生成前就绑定肯定是无效的,因为所绑定的元素目前根本不存在. 然而, ...

  2. js字符串与十六进制之间的转换

    在寻找加密解密的时候看到一个方法,代码图片转换.原理为:字符可以转为16进制,与图片RGB的一个R/G/B相对应,即一个像素点可容纳3个字符(注:Canvas的RGBA,透明度A似乎不能使用,使用后, ...

  3. Ubuntu获取root 权限,开机自动登入root

    新机器获取root权限,只需要给root 增加密码: sudo passwd root 修改开机自动登入: #sudo gedit /etc/lightdm/lightdm.conf 修改参数: au ...

  4. ios 12 xcode10 新升级的编译报错libstdc++.6.0.9 Multiple commands produce

    问题一 编译报错 Showing Recent Messages :-1: Multiple commands produce '/Users/duning/Library/Developer/Xco ...

  5. PHP 解决同一个IP不同端口号session冲突的问题

    在项目的开发阶段,我们经常会遇到几个站点共用同一个IP用不同端口号区分的形式!但是,这样很容易导致一个问题,session冲突丢失!即两个站点具有相同的session变量,清除session的时候即全 ...

  6. SCCM Collection 集合获取计算机最后启动时间

    获取计算机客户端最后一次启动时间,我们可以通过多种来源获取,如活动目录组 ,而不仅仅是SCCM 收集,希望对您有所帮助,下面分享PowerShell 脚本 # 1 $CollectionName = ...

  7. vue.js的package.json相关问题解惑

    使用vue-cli创建vue.webpack项目,在项目中用到了iSlider移动端滑动插件,只在本地命令工具中npm install islider.js:提交之后,partner下载代码后一直运行 ...

  8. javaSe-线程

    package com.java.chap09.sec02; public class Thread1 extends Thread{ private int baoZi=1; private Str ...

  9. Java 文件操作-RandomAccessFile

    1. RandomAccessFile     Java提供了一个可以对文件随机访问的操作,访问包括读和写操作.该类名为RandomAccessFile.该类的读写是基于指针的操作. 1)文件访问模式 ...

  10. 程序员的智囊库系列之2----网站框架(framework)

    程序员的智囊库系列之2--网站框架(framework) 这是程序员的智囊库系列的第二篇文章.上一篇文章讲了服务器与运维相关的工具,这篇文章我们将介绍几个搭建网站的框架: django express ...