#include ”stdio.h“ 
#include “conio.h” 
#include ”assert.h“ 
#include “stdlib.h” 
#define MAPMAXSIZE 100 //地图面积最大为 100x100 
#define MAXINT 8192 //定义一个最大整数, 地图上任意两点距离不会超过它 
#define STACKSIZE 65536 //保存搜索节点的堆栈大小

#define tile_num(x,y) ((y)*map_w+(x)) //将 x,y 坐标转换为地图上块的编号 
#define tile_x(n) ((n)%map_w) //由块编号得出 x,y 坐标 
#define tile_y(n) ((n)/map_w)

// 树结构, 比较特殊, 是从叶节点向根节点反向链接 
typedef struct node *TREE;

struct node { 
int h; 
int tile; 
TREE father; 
} ;

typedef struct node2 *LINK;

struct node2 { 
TREE node; 
int f; 
LINK next; 
};

LINK queue; // 保存没有处理的行走方法的节点 
TREE stack[STACKSIZE]; // 保存已经处理过的节点 (搜索完后释放) 
int stacktop; 
unsigned char map[MAPMAXSIZE][MAPMAXSIZE]; //地图数据 
int dis_map[MAPMAXSIZE][MAPMAXSIZE]; //保存搜索路径时,中间目标地最优解

int map_w,map_h; //地图宽和高 
int start_x,start_y,end_x,end_y; //地点,终点坐标

// 初始化队列 
void init_queue() 

queue=(LINK)malloc(sizeof(*queue)); 
queue->node=NULL; 
queue->f=-1; 
queue->next=(LINK)malloc(sizeof(*queue)); 
queue->next->f=MAXINT; 
queue->next->node=NULL; 
queue->next->next=NULL; 
}

// 待处理节点入队列, 依靠对目的地估价距离插入排序 
void enter_queue(TREE node,int f) 

LINK p=queue,father,q; 
while(f>p->f) { 
father=p; 
p=p->next; 
assert(p); 

q=(LINK)malloc(sizeof(*q)); 
assert(queue); 
q->f=f,q->node=node,q->next=p; 
father->next=q; 
}

// 将离目的地估计最近的方案出队列 
TREE get_from_queue() 

TREE bestchoice=queue->next->node; 
LINK next=queue->next->next; 
free(queue->next); 
queue->next=next; 
stack[stacktop++]=bestchoice; 
assert(stacktop<STACKSIZE); 
return bestchoice; 
}

// 释放栈顶节点 
void pop_stack() 

free(stack[--stacktop]); 
}

// 释放申请过的所有节点 
void freetree() 

int i; 
LINK p; 
for (i=0;i<stacktop;i++) 
free(stack); 
while (queue) { 
p=queue; 
free(p->node); 
queue=queue->next; 
free(p); 

}

// 估价函数,估价 x,y 到目的地的距离,估计值必须保证比实际值小 
int judge(int x,int y) 

int distance; 
distance=abs(end_x-x)+abs(end_y-y); 
return distance; 
}

// 尝试下一步移动到 x,y 可行否 
int trytile(int x,int y,TREE father) 

TREE p=father; 
int h; 
if (map[y][x]!=' ') return 1; // 如果 (x,y) 处是障碍,失败 
while (p) { 
if (x==tile_x(p->tile) && y==tile_y(p->tile)) return 1; //如果 (x,y) 曾经经过,失败 
p=p->father; 

h=father->h+1; 
if (h>=dis_map[y][x]) return 1; // 如果曾经有更好的方案移动到 (x,y) 失败 
dis_map[y][x]=h; // 记录这次到 (x,y) 的距离为历史最佳距离

// 将这步方案记入待处理队列 
p=(TREE)malloc(sizeof(*p)); 
p->father=father; 
p->h=father->h+1; 
p->tile=tile_num(x,y); 
enter_queue(p,p->h+judge(x,y)); 
return 0; 
}

// 路径寻找主函数 
void findpath(int *path) 

TREE root; 
int i,j; 
stacktop=0; 
for (i=0;i<map_h;i++) 
for (j=0;j<map_w;j++) 
dis_map[j]=MAXINT; 
init_queue(); 
root=(TREE)malloc(sizeof(*root)); 
root->tile=tile_num(start_x,start_y); 
root->h=0; 
root->father=NULL; 
enter_queue(root,judge(start_x,start_y)); 
for (;;) { 
int x,y,child; 
TREE p; 
root=get_from_queue(); 
if (root==NULL) { 
*path=-1; 
return; 

x=tile_x(root->tile); 
y=tile_y(root->tile); 
if (x==end_x && y==end_y) break; // 达到目的地成功返回

child=trytile(x,y-1,root); //尝试向上移动 
child&=trytile(x,y+1,root); //尝试向下移动 
child&=trytile(x-1,y,root); //尝试向左移动 
child&=trytile(x+1,y,root); //尝试向右移动 
if (child!=0) 
pop_stack(); // 如果四个方向均不能移动,释放这个死节点 
}

// 回溯树,将求出的最佳路径保存在 path[] 中 
for (i=0;root;i++) { 
path=root->tile; 
root=root->father; 

path=-1; 
freetree(); 
}

void printpath(int *path) 

int i; 
for (i=0;path>=0;i++) { 
gotoxy(tile_x(path)+1,tile_y(path)+1); 
cprintf("\xfe"); 

}

int readmap() 

FILE *f; 
int i,j; 
f=fopen("map.dat","r"); 
assert(f); 
fscanf(f,"%d,%d\n",&map_w,&map_h); 
for (i=0;i<map_h;i++) 
fgets(&map[0],map_w+1,f); 
fclose(f); 
start_x=-1,end_x=-1; 
for (i=0;i<map_h;i++) 
for (j=0;j<map_w;j++) { 
if (map[j]=='s') map[j]=' ',start_x=j,start_y=i; 
if (map[j]=='e') map[j]=' ',end_x=j,end_y=i; 

assert(start_x>=0 && end_x>=0); 
return 0; 
}

void showmap() 

int i,j; 
clrscr(); 
for (i=0;i<map_h;i++) { 
gotoxy(1,i+1); 
for (j=0;j<map_w;j++) 
if (map[j]!=' ') cprintf("\xdb"); 
else cprintf(" "); 

gotoxy(start_x+1,start_y+1); 
cprintf("s"); 
gotoxy(end_x+1,end_y+1); 
cprintf("e"); 
}

int main() 

int path[MAXINT]; 
readmap(); 
showmap(); 
getch(); 
findpath(path); 
printpath(path); 
getch(); 
return 0; 
}

A*算法的C语言实现的更多相关文章

  1. 魔方阵算法及C语言实现

    1 魔方阵概念 填充的,每一行.每一列.对角线之和均相等的方阵,阶数n = 3,4,5….魔方阵也称为幻方阵. 例如三阶魔方阵为: 魔方阵有什么的规律呢? 魔方阵分为奇幻方和偶幻方.而偶幻方又分为是4 ...

  2. 一个UUID生成算法的C语言实现 --- WIN32版本 .

    一个UUID生成算法的C语言实现——WIN32版本   cheungmine 2007-9-16   根据定义,UUID(Universally Unique IDentifier,也称GUID)在时 ...

  3. 无限大整数相加算法的C语言源代码

    忙里偷闲,终于完成了无限大整数相加算法的C语言代码,无限大整数相加算法的算法分析在这里. 500位的加法运行1000次,不打印结果的情况下耗时0.036秒,打印结果的情况下耗时16.285秒. 下面是 ...

  4. 数据结构算法集---C++语言实现

    //数据结构算法集---C++语言实现 //各种类都使用模版设计,可以对各种数据类型操作(整形,字符,浮点) /////////////////////////// // // // 堆栈数据结构 s ...

  5. 1164: 零起点学算法71——C语言合法标识符(存在问题)

    1164: 零起点学算法71——C语言合法标识符 Time Limit: 1 Sec  Memory Limit: 64 MB   64bit IO Format: %lldSubmitted: 10 ...

  6. 【最全】经典排序算法(C语言)

    算法复杂度比较: 算法分类 一.直接插入排序 一个插入排序是另一种简单排序,它的思路是:每次从未排好的序列中选出第一个元素插入到已排好的序列中. 它的算法步骤可以大致归纳如下: 从未排好的序列中拿出首 ...

  7. PID算法(c 语言)(转)

    PID算法(c 语言)(来自老外) #include <stdio.h> #include<math.h> //定义PID 的结构体 struct _pid { int pv; ...

  8. 一个UUID生成算法的C语言实现——WIN32版本

    源: 一个UUID生成算法的C语言实现——WIN32版本

  9. 排序算法总结(C语言版)

    排序算法总结(C语言版) 1.    插入排序 1.1     直接插入排序 1.2     Shell排序 2.    交换排序 2.1     冒泡排序 2.2     快速排序 3.    选择 ...

  10. matlab算法转为c语言注意事项

    matlab算法转为c语言后,影响c语言效率的关键在于multiword的产生,基于此会有multiword加减法和乘除法,极大消耗资源,减少甚至消除multiword很重要,需注意的是:算法中尽量减 ...

随机推荐

  1. Apache proxy中转设置

    参考http://sjsky.iteye.com/blog/1067119 打开http.conf  (macOS中 Apache配置文件在/etc/apache2/中   etc是隐藏的) 确保下面 ...

  2. yoeman构建Asp.net core项目并且实现分层

    在Mac上开发使用yoeman构建Asp.net core项目并且实现分层引用 1.Yoeman? yoeman是一个自动化脚手架工具.它提供很多generator,generator相当于Visua ...

  3. Oracle SQL篇(一)null值之初体验

           从我第一次正式的写sql语句到现在,已经超过10年的时间了.我写报表,做统计分析和财务对账,我一点点的接触oracle数据库,并尝试深入了解.这条路,一走就是10年,从充满热情,到开始厌 ...

  4. QT实现TCP通信服务器端和客户端(支持多个客户端)精简版

    上星期接了个私活,工期两星期,报酬3000,写一个小软件,采集定向网络上的数据,并进行双向通信,捣鼓了两天,终于把QT中tcp通信这块调通了,找过N多例子,绝大部分都是基本的一个服务端一个客户端通信的 ...

  5. DLL与EXE之间的通讯调用 以及 回调函数的线程执行空间

    dll 与 exe 之间的通讯方式有很多种, 本文采用回调函数的方法实现, 本文也将研究多线程,多模块的情况下,回调函数所在的线程, 啥也不说了,先附上代码: 下面的是dll模块的的, dll的工程文 ...

  6. dump文件定位程序崩溃代码行

    1.dump文件 2.程序对应的pdb 步骤一:安装windbg 步骤二:通过windbg打开crash dump文件 步骤三:设置pdb文件路径,即符号表路径 步骤四:运行命令!analyze -v ...

  7. 用户管理_组管理_设置主机名_UGO_文件高级权限_ACL权限

    用户管理: 添加用户:useradd tom 设置密码:passwd tom 切换账户: su - tom (不加-也能切换,但是 -会有两点不同 1.有-会切换到该用户的主目录  2.会切换到该用户 ...

  8. elk 分布式数据同步

    zjtest7-redis:/elk/elasticsearch/data/es_cluster/nodes/0/indices/library# strings ./1/index/_3.cfs | ...

  9. elasticsearch集群部署

    启动elk: zjtest7-redis:/usr/local/elasticsearch-2.3.4/bin# ./elasticsearch -d 后台运行 访问: http://192.168. ...

  10. liquibase之快速入门

    第一步: 创建一个Changelog File: 这个database  Changelog file列举了数据库中所有的改变情况,该文件是以xml为基础的,下面是一个空的xml文件: <?xm ...