青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

無我

讓內(nèi)心永遠(yuǎn)燃燒著偉大的光明的精神之火!
靈活的思考,嚴(yán)謹(jǐn)?shù)膶?shí)現(xiàn)
豪邁的氣魄、頑強(qiáng)的意志和周全的思考

【轉(zhuǎn)】優(yōu)化后的A*算法


優(yōu)化后的A*算法--簡潔、易懂且實(shí)用
添加時(shí)間:2006-12-15   出處:互聯(lián)網(wǎng)  作者:zhangshuo
 
  我曾看過一些有關(guān)A*算法的程序,不過寫得比較簡潔、易懂的還是風(fēng)云寫的A*算法教學(xué)實(shí)例(風(fēng)云工作室),但是這個(gè)算法并沒有進(jìn)行優(yōu)化,該程序要用到實(shí)際應(yīng)用中,還會(huì)有一定的限制,所以我對(duì)該算法進(jìn)行了改進(jìn),并加上更詳細(xì)的算法說明,使其具有更好的教學(xué)作用和實(shí)用價(jià)值。開始前我先給出A*算法的基本思路:
  問題:求出2D的迷宮中起始點(diǎn)S到目標(biāo)點(diǎn)E的最短路徑?
  算法: findpath()
  {
  把S點(diǎn)加入樹根(各點(diǎn)所在的樹的高度表示從S點(diǎn)到該點(diǎn)所走過的步數(shù));
  把S點(diǎn)加入排序隊(duì)列(按該點(diǎn)到E點(diǎn)的距離排序+走過的步數(shù)從小到大排序);
  1、排序隊(duì)列sort_queue中距離最小的第一個(gè)點(diǎn)出列,并保存入store_queue中
  2、從出列的點(diǎn)出發(fā),分別向4個(gè)(或8個(gè))方向中的一個(gè)各走出一步
  3、并估算第2步所走到位置到目標(biāo)點(diǎn)的距離,并把該位置加入樹,
  最后把該點(diǎn)按距離從小到大排序后并放入隊(duì)列中。(由trytile函數(shù)實(shí)現(xiàn))。
  4、如果該點(diǎn)從四個(gè)方向上都不能移動(dòng),則把該點(diǎn)從store_queue中刪除
  5、回到第一點(diǎn),直到找到E點(diǎn)則結(jié)束
  從目標(biāo)點(diǎn)回溯樹,直到樹根則可以找到最佳路徑,并保存在path[]中
  }
  /*========================================================================
  精簡的A*算法 作者:添翼虎
  網(wǎng)址:http://tyhweb.163.net Email:tyhweb@163.net
  本程序參考了風(fēng)云的最短路徑代碼(http://member.nease.com/~cloudwu),
  并加以改進(jìn)和優(yōu)化:
  1、把原來用于存放已處理節(jié)點(diǎn)的堆棧改為(store_queue)隊(duì)列,這樣在從
  sort_queue隊(duì)列出列時(shí)可直接放入store_queue中。
  2、解除了地圖大小的限制(如果有64K內(nèi)存限制時(shí),地圖大小只能是180x180)
  3、刪除了原程序中的一些冗余,見程序中的注釋。
  4、程序繼續(xù)使用dis_map數(shù)組保存各點(diǎn)歷史歷史最佳距離,也包含了某點(diǎn)是否已經(jīng)
  經(jīng)過的信息,雖然這樣做可能會(huì)比使用鏈表多用一些內(nèi)存,但是在搜索時(shí)可以
  節(jié)省不時(shí)間。
  5、程序更具有實(shí)用性,可直接或修改后運(yùn)用于你的程序中,但請(qǐng)你使用該代碼后
  應(yīng)該返回一些信息給我,如算法的改進(jìn)或使用于什么程序等。
  本程序可以用Borland C++或DJGPP編譯,并附帶有一個(gè)數(shù)據(jù)文件 map.dat,
  保存有地圖的數(shù)據(jù),(注:該地圖文件格式與風(fēng)云的原代碼的地圖格式不一樣)
  -------------------------------------------------------------------------*/
  //#define NDEBUG
  #include <stdio.h>
  #include <conio.h>
  #include <assert.h>
  #include <stdlib.h>
  
  #define tile_num(x,y) ((y)*map_w+(x)) //將 x,y 坐標(biāo)轉(zhuǎn)換為地圖上塊的編號(hào)
  #define tile_x(n) ((n)%map_w) //由塊編號(hào)得出 x,y 坐標(biāo)
  #define tile_y(n) ((n)/map_w)
  
  #define MAPMAXSIZE 180 //地圖面積最大為 180x180,如果沒有64K內(nèi)存限制可以更大
  #define MAXINT 32767
  //樹結(jié)構(gòu), 比較特殊, 是從葉節(jié)點(diǎn)向根節(jié)點(diǎn)反向鏈接,方便從葉節(jié)點(diǎn)找到根節(jié)點(diǎn)
  typedef struct tree_node *TREE;
  struct tree_node {
  int h; //節(jié)點(diǎn)所在的高度,表示從起始點(diǎn)到該節(jié)點(diǎn)所有的步數(shù)
  int tile; //該節(jié)點(diǎn)的位置
  TREE father; //該節(jié)點(diǎn)的上一步
  };
  //鏈接結(jié)構(gòu),用于保存處理過的和沒有處理過的結(jié)點(diǎn)
  typedef struct link_node *LINK;
  struct link_node {
  TREE node;
  int f;
  LINK next;
  };
  LINK sort_queue; // 保存沒有處理的行走方法的節(jié)點(diǎn)
  LINK store_queue; // 保存已經(jīng)處理過的節(jié)點(diǎn) (搜索完后釋放)
  unsigned char * map; //地圖數(shù)據(jù)
  unsigned int * dis_map; //保存搜索路徑時(shí),中間目標(biāo)地最優(yōu)解
  int map_w,map_h; //地圖寬和高
  int start_x,start_y,end_x,end_y; //地點(diǎn),終點(diǎn)坐標(biāo)
  // 初始化隊(duì)列
  void init_queue()
  {
  sort_queue=(LINK)malloc(sizeof(*sort_queue));
  sort_queue->node=NULL;
  sort_queue->f=-1;
  sort_queue->next=(LINK)malloc(sizeof(*sort_queue));
  sort_queue->next->node=NULL;
  sort_queue->next->f=MAXINT;
  sort_queue->next->next=NULL;
  store_queue=(LINK)malloc(sizeof(*store_queue));
  store_queue->node=NULL;
  store_queue->f=-1;
  store_queue->next=NULL;
  }
  // 待處理節(jié)點(diǎn)入隊(duì)列, 依靠對(duì)目的地估價(jià)距離插入排序
  void enter_queue(TREE node,int f)
  {
  LINK p=sort_queue,father,q;
  while(f>p->f) {
  father=p;
  p=p->next;
  assert(p);
  }
  q=(LINK)malloc(sizeof(*q));
  assert(sort_queue);
  q->f=f,q->node=node,q->next=p;
  father->next=q;
  }
  // 將離目的地估計(jì)最近的方案出隊(duì)列
  TREE get_from_queue()
  {
  LINK bestchoice=sort_queue->next;
  LINK next=sort_queue->next->next;
  sort_queue->next=next;
  bestchoice->next=store_queue->next;
  store_queue->next=bestchoice;
  return bestchoice->node;
  }
  // 釋放棧頂節(jié)點(diǎn)
  void pop_stack()
  {
  LINK s=store_queue->next;
  assert(s);
  store_queue->next=store_queue->next->next;
  free(s->node);
  free(s);
  }
  // 釋放申請(qǐng)過的所有節(jié)點(diǎn)
  void freetree()
  {
  int i;
  LINK p;
  while(store_queue){
  p=store_queue;
  free(p->node);
  store_queue=store_queue->next;
  free(p);
  }
  while (sort_queue) {
  p=sort_queue;
  free(p->node);
  sort_queue=sort_queue->next;
  free(p);
  }
  }
  // 估價(jià)函數(shù),估價(jià) x,y 到目的地的距離,估計(jì)值必須保證比實(shí)際值小
  int judge(int x,int y)
  {
  int distance;
  distance=abs(end_x-x)+abs(end_y-y);
  return distance;
  }
  // 嘗試下一步移動(dòng)到 x,y 可行否
  int trytile(int x,int y,TREE father)
  {
  TREE p=father;
  int h;
  if (map[tile_num(x,y)]!=' ') return 1; // 如果 (x,y) 處是障礙,失敗
  //這一步用來判斷(x,y)點(diǎn)是否已經(jīng)加入隊(duì)列,多余可以刪除,因?yàn)閐is_map已經(jīng)
  //保存該點(diǎn)是否已經(jīng)保存
  //while (p) {
  // if (x==tile_x(p->tile) && y==tile_y(p->tile)) return 1; //如果 (x,y) 曾經(jīng)經(jīng)過,失敗
  // p=p->father;
  //}
  h=father->h+1;
  if (h>=dis_map[tile_num(x,y)]) return 1; // 如果曾經(jīng)有更好的方案移動(dòng)到 (x,y) 失敗
  dis_map[tile_num(x,y)]=h; // 記錄這次到 (x,y) 的距離為歷史最佳距離
  // 將這步方案記入待處理隊(duì)列
  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;
  }
  // 路徑尋找主函數(shù)
  int * findpath(void)
  {
  TREE root;
  int i,j;
  int * path;
  memset(dis_map,0xff,map_h*map_w*sizeof(*dis_map)); //填充dis_map為0XFF,表示各點(diǎn)未曾經(jīng)過
  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) {
  return NULL;
  }
  x=tile_x(root->tile);
  y=tile_y(root->tile);
  if (x==end_x && y==end_y) break; // 達(dá)到目的地成功返回
  child=trytile(x,y-1,root); //嘗試向上移動(dòng)
  child&=trytile(x,y+1,root); //嘗試向下移動(dòng)
  child&=trytile(x-1,y,root); //嘗試向左移動(dòng)
  child&=trytile(x+1,y,root); //嘗試向右移動(dòng)
  //child&=trytile(x+1,y-1,root);//嘗試向右上移動(dòng)
  //child&=trytile(x+1,y+1,root); //嘗試向右下移動(dòng)
  //child&=trytile(x-1,y+1,root); //嘗試向左下移動(dòng)
  //child&=trytile(x-1,y-1,root); //嘗試向左上移動(dòng)
  if (child!=0)
  pop_stack(); // 如果四個(gè)方向均不能移動(dòng),釋放這個(gè)死節(jié)點(diǎn)
  }
  // 回溯樹,將求出的最佳路徑保存在 path[] 中
  path=(int*)malloc((root->h+2)*sizeof(int));
  assert(path);
  for (i=0;root;i++) {
  path[i]=root->tile;
  root=root->father;
  }
  path[i]=-1;
  freetree();
  return path;
  }
  void printpath(int *path)
  {
  int i;
  if(path==NULL) return ;
  for (i=0;path[i]>=0;i++) {
  gotoxy(tile_x(path[i])+1,tile_y(path[i])+1);
  cprintf(".");
  }
  }
  int readmap()
  {
  FILE *f;
  int i,j;
  f=fopen("map.dat","r");
  assert(f);
  fscanf(f,"%d,%d\n",&map_w,&map_h);
  map=malloc(map_w*map_h+1);
  assert(map);
  for(i=0;i<map_h;i++)


 

posted on 2007-07-22 15:42 Tim 閱讀(1132) 評(píng)論(0)  編輯 收藏 引用 所屬分類: 數(shù)據(jù)結(jié)構(gòu)和算法


只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


<2007年7月>
24252627282930
1234567
891011121314
15161718192021
22232425262728
2930311234

導(dǎo)航

統(tǒng)計(jì)

公告

本博客原創(chuàng)文章,歡迎轉(zhuǎn)載和交流。不過請(qǐng)注明以下信息:
作者:TimWu
郵箱:timfly@yeah.net
來源:www.shnenglu.com/Tim
感謝您對(duì)我的支持!

留言簿(9)

隨筆分類(173)

IT

Life

搜索

積分與排名

最新隨筆

最新評(píng)論

閱讀排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美日韩国产色视频| 国产欧美日韩伦理| 亚洲破处大片| 亚洲黄色免费电影| 欧美.com| 亚洲午夜激情网页| 亚洲男人的天堂在线aⅴ视频| 国产精品夜夜夜| 久久亚裔精品欧美| 美女视频一区免费观看| 在线视频你懂得一区二区三区| 一区二区三区日韩在线观看| 国产精品视频久久一区| 久久亚洲一区二区三区四区| 免费日本视频一区| 亚洲免费一区二区| 久久久久久一区| 亚洲天堂网站在线观看视频| 欧美一区二区观看视频| 亚洲黄色片网站| 亚洲图片欧美午夜| 91久久夜色精品国产九色| 亚洲午夜在线| 91久久久精品| 欧美在线观看视频一区二区| 91久久久久久| 午夜精品影院| 99视频精品免费观看| 欧美一区免费视频| 99国产成+人+综合+亚洲欧美| 午夜视黄欧洲亚洲| 日韩视频在线观看国产| 销魂美女一区二区三区视频在线| 亚洲日韩欧美视频一区| 欧美一区二区黄色| 中文国产亚洲喷潮| 美玉足脚交一区二区三区图片| 亚洲一级黄色av| 免费亚洲一区二区| 久久久久久69| 国产精品另类一区| 亚洲激情图片小说视频| 国外成人性视频| 亚洲无限av看| 亚洲一二三区在线观看| 欧美激情成人在线视频| 久久青草欧美一区二区三区| 国产精品久久久久9999吃药| 亚洲国产美女| 亚洲第一区中文99精品| 欧美一级黄色录像| 亚洲欧美制服中文字幕| 欧美韩国日本综合| 亚洲电影欧美电影有声小说| 国产一区日韩欧美| 亚洲欧美日韩中文在线制服| 亚洲综合欧美日韩| 欧美日韩亚洲激情| 亚洲精品影院在线观看| 最新亚洲视频| 欧美成人午夜剧场免费观看| 欧美暴力喷水在线| 亚洲国产三级| 欧美国产高潮xxxx1819| 亚洲欧洲日产国产综合网| 亚洲精品老司机| 欧美激情偷拍| 日韩午夜在线| 亚洲免费在线视频一区 二区| 欧美日韩国产系列| 中日韩男男gay无套| 午夜精品一区二区三区在线播放| 国产精品白丝jk黑袜喷水| 亚洲视频欧美视频| 欧美在线影院| 韩国成人精品a∨在线观看| 久久精品国产亚洲高清剧情介绍| 久久精品动漫| 91久久精品国产| 欧美精品www| 亚洲网站视频| 老司机免费视频一区二区| 1024亚洲| 欧美日韩妖精视频| 性色一区二区三区| 蜜臀99久久精品久久久久久软件| 亚洲国产网站| 国产精品国产a级| 久久riav二区三区| 亚洲国产精品va在线看黑人 | 久久精品一区中文字幕| 美女在线一区二区| 在线亚洲自拍| 狠狠色香婷婷久久亚洲精品| 久久综合网络一区二区| 亚洲精品久久久久久久久| 性做久久久久久久久| 1000部精品久久久久久久久| 欧美人与性禽动交情品| 午夜精品视频在线观看一区二区| 美女图片一区二区| 亚洲一二区在线| 亚洲国产欧美一区二区三区丁香婷 | 99视频+国产日韩欧美| 国产精品乱人伦一区二区| 久久女同互慰一区二区三区| 一二三区精品| 欧美成人免费大片| 香蕉久久夜色精品国产| 亚洲日本激情| 韩国精品在线观看| 国产精品免费aⅴ片在线观看| 久久天天躁狠狠躁夜夜av| 亚洲狼人综合| 欧美激情精品久久久久久大尺度| 午夜电影亚洲| 亚洲视频香蕉人妖| 亚洲国产欧美不卡在线观看| 国产日本亚洲高清| 欧美日韩一区视频| 麻豆精品一区二区av白丝在线| 亚洲欧美国产一区二区三区| 亚洲日本视频| 免费成人在线视频网站| 久久成人一区二区| 午夜在线播放视频欧美| 在线视频日韩| 日韩香蕉视频| 亚洲精品偷拍| 亚洲欧洲综合另类| 91久久精品国产| 亚洲国产精品久久久久婷婷884| 国产女精品视频网站免费| 欧美视频手机在线| 欧美日韩国产一中文字不卡| 欧美福利小视频| 欧美sm视频| 欧美xxx成人| 欧美福利在线| 欧美激情第三页| 欧美精品18| 欧美日本一区二区三区| 欧美69视频| 欧美极品在线播放| 欧美日韩国产bt| 欧美视频精品在线| 国产精品人人做人人爽人人添| 欧美日韩国产经典色站一区二区三区| 欧美α欧美αv大片| 欧美高潮视频| 欧美午夜宅男影院在线观看| 欧美日韩中文精品| 国产精品毛片一区二区三区| 国产精品日韩电影| 国产一区二区三区免费观看| 狠狠做深爱婷婷久久综合一区| 韩日欧美一区二区三区| 在线观看日韩av电影| 亚洲人久久久| 亚洲午夜久久久久久久久电影院 | 久久av老司机精品网站导航| 久久av一区二区三区| 久久亚洲风情| 亚洲国产欧美一区二区三区同亚洲| 亚洲大胆视频| 亚洲视频成人| 久久久亚洲国产天美传媒修理工 | 久久亚洲综合网| 欧美日本在线一区| 国产精品永久免费观看| 一区二区三区在线免费播放| 亚洲精品在线观| 性欧美长视频| 欧美国产日韩一区二区三区| 亚洲精品免费网站| 亚洲欧美日韩国产综合在线| 久久精品一区中文字幕| 欧美精品不卡| 好吊色欧美一区二区三区四区 | 日韩一级精品视频在线观看| 欧美一区二区高清| 亚洲国产成人在线| 欧美亚洲在线播放| 欧美精品在线一区二区| 国产视频久久久久久久| 亚洲日本aⅴ片在线观看香蕉| 午夜国产不卡在线观看视频| 欧美va天堂| 亚洲综合色婷婷| 欧美日韩国语| 亚洲电影激情视频网站| 欧美一级日韩一级| 亚洲激情网站| 久久亚洲综合网| 国产欧美一区二区三区视频| 日韩一级网站| 欧美刺激午夜性久久久久久久| 午夜精品一区二区三区四区| 欧美日韩精品免费观看视频| 尤物九九久久国产精品的分类|