- 浏览: 1088510 次
- 性别:
- 来自: 杭州
文章分类
- 全部博客 (695)
- 心情日记 (14)
- AS开发工具 (12)
- 文章转载 (99)
- AIR (5)
- 问题总结 (46)
- SWF格式 (7)
- 测试总结 (10)
- 外文资料 (9)
- 算法技术 (33)
- AS3常用开源库 (43)
- 源码范例 (102)
- FLEX (72)
- FLASH 优化 (33)
- 游戏开发 (49)
- 开发技术 (11)
- 工作应用 (34)
- AS3收集 (140)
- WebBase (0)
- 开发构想 (4)
- 设计模式 (2)
- 框架和框架范例 (19)
- RED5 (3)
- java开发 (3)
- JAVA (1)
- FLASH-3D (23)
- 3D (6)
- 书籍 (10)
- 业界信息资料 (3)
- C# (1)
- JavaScript (12)
- HTML5 (6)
- Flixel (1)
- D5Power RPG网页游戏引擎 (0)
- ColorMatrixFilter - 获得相应颜色的色调 函数 (0)
- Starling (0)
最新评论
-
老顽童203:
字体
水果忍者鼠标跟随特效制作[转载] -
hairball00:
[转] 放出超多的Flash组件源代码 -
he74552775:
flash AS3 RegExp简单功能用法(转) -
hanshuai1232000:
第四点,有利也有弊,等你做了大型的aprg,你就知道了
[转]位图数据内存优化 -
yangfantao:
太感谢
[转] 放出超多的Flash组件源代码
Introduction to Heuristics Search------
A* Algorithm Principle and Practice
[Classify] Algorithm Implementation , Artificial Intelligence
[ Level ] ★★★★★
[Abstract] This article introduces an important and effective heuristics algorithm ------ A* algorithm with its theory and also gives out its interactive implementation of path finding problem.
[Key Words] A* , Heuristics Algorithm , Best Path Finding , Interactive , AS2
[1历史回顾]
P. E. Hart , N. J. Nilsson 和B. Raphael共同发表了一篇在启发式搜索方面有深远影响力的论文:
“P. E. Hart, N. J. Nilsson, and B. Raphael. A formal basis for the heuristic determination of minimum cost paths in graphs. IEEE Trans. Syst. Sci. and Cybernetics, SSC-4(2):100-107, 1968”。从此,一种精巧、高效的算法------A*算法横空出世了,并在相关领域得到了广泛的应用。
Peter E. Hart
Nils J. Nilsson
[2从DFS和BFS中来]
A*算法的思想来源并不是什么高深莫测的东西,事实上,它与我们所熟悉的另外两种搜索策略:DFS(深度优先Deep First Search)和 BFS(广度优先Breadth First Search)有着自然而紧密的联系。
首先要提一下搜索树的概念,一个可以搜索出某个可行解的问题,如“农夫、白菜、羊、狼”和“八皇后”等,虽然从表面上看上去和“树”这种结构无关,但是整个搜索过程中的可能试探点所行成的搜索空间总可以对应到一颗搜索树上去。所以,将各类形式上不同的搜索问题抽象并统一成为搜索树的形式,为算法的设计与分析带来巨大的方便。
下面,让我们通过两个Flash演示来回顾一下DFS和BFS的算法思想:
DFS算法思想演示:
http://www.ogdev.net/download/2006/deep.swf
BFS算法思想演示:
http://www.ogdev.net/download/2006/breadth.swf
DFS和BFS在展开子结点时均属于盲目型搜索,也就是说,它不会选择哪个结点在下一次搜索中更优而去跳转到该结点进行下一步的搜索。在运气不好的情形中,均需要试探完整个解集空间, 显然,只能适用于问题规模不大的搜索问题中。
那么,作为启发式算法中的A*算法,又比它们高效在哪里呢?
首先要来谈一下什么是启发式算法。所谓启发式搜索,与DFS和BFS这类盲目型搜索最大的不同,就在于当前搜索结点往下选择下一步结点时,可以通过一个启发函数来进行选择,选择代价最少的结点作为下一步搜索结点而跳转其上(遇到有一个以上代价最少的结点,不妨选距离当前搜索点最近一次展开的搜索点进行下一步搜索)。一个经过仔细设计的启发函数,往往在很快的时间内就可得到一个搜索问题的最优解,对于NP问题,亦可在多项式时间内得到一个较优解。
A*算法,作为启发式算法中很重要的一种,被广泛应用在最优路径求解和一些策略设计的问题中。而A*算法最为核心的部分,就在于它的一个估值函数的设计上:
f(n)=g(n)+h(n)
其中f(n)是每个可能试探点的估值,它有两部分组成:一部分为g(n),它表示从起始搜索点到当前点的代价(通常用某结点在搜索树中的深度来表示)。另一部分,即h(n),它表示启发式搜索中最为重要的一部分,即当前结点到目标结点的估值,h(n)设计的好坏,直接影响着具有此种启发式函数的启发式算法的是否能称为A*算法。
一种具有f(n)=g(n)+h(n)策略的启发式算法能成为A*算法的充分条件是:
1) 搜索树上存在着从起始点到终了点的最优路径。
2) 问题域是有限的。
3)所有结点的子结点的搜索代价值>0。
4)h(n)=
当此四个条件都满足时,一个具有f(n)=g(n)+h(n)策略的启发式算法能成为A*算法,并一定能找到最优解。([1]P89给出了相关的证明)
对于一个搜索问题,显然,条件1,2,3都是很容易满足的,而
条件4): h(n)<=h*(n)是需要精心设计的,由于h*(n)显然是无法知道的,
所以,一个满足条件4)的启发策略h(n)就来的难能可贵了。不过,对于图的最优路径搜索和八数码问题,有些相关策略h(n)不仅很好理解,而且已经在理论上证明是满足条件4)的,从而为这个算法的推广起到了决定性的作用。不过h(n)距离h*(n)的呈度不能过大,否则h(n)就没有过强的区分能力,算法效率并不会很高。对一个好的h(n)的评价是:h(n)在h*(n)的下界之下,并且尽量接近h*(n).
当然,估值函数的设计也就就仅仅是f(n)=g(n)+h(n)一种,另外的估值函数“变种”如:f(n)=w*g(n)+(1-w)*h(n) ,f(n)=g(n)+h(n)+h(n-1)针对不同的具体问题亦会有不同的效果。
[3A*算法的核心过程]
让我们先来通过两个Flash演示,来看一下A*算法在具体应用时的搜索树是如何展开的。
http://www.ogdev.net/download/2006/searchsample1.swf
A* Search搜索树1
http://www.ogdev.net/download/2006/searchsample1.swf
A* Search搜索树2
通过演示,我们可以看出:A*算法最为核心的过程,就在每次选择下一个当前搜索点时,是从所有已探知的但未搜索过点中(可能是不同层,亦可不在同一条支路上),选取f值最小的结点进行展开。而所有“已探知的但未搜索过点”可以通过一个按f值升序的队列(即优先队列)进行排列。这样,在整体的搜索过程中,只要按照类似广度优先的算法框架,从优先队列中弹出队首元素(f值),对其可能子结点计算g、h和f值,直到优先队列为空(无解)或找到终止点为止。
A*算法与广度优先和深度优先的联系就在于,当g(n)=0时,该算法类似于DFS,当h(n)=0时,该算法类似于BFS , 这一点,可以通过上面的A*搜索树的具体过程中将h(n)设为0或将g(n)设为0而得到。
A*算法实现框架:
重要数据解释:
Open Table :存放所有已探知的但未搜索过点的优先队列。
Closed Table :存在搜索过的点的数组,提取最优路径时有用。
Start Node :起始点。
Target Node :终止点。
C Node :当前点。
算法框架如下:
1.Init start node , add it to open table
While not reach target node && open table is unNull
2.a) Get the head node in open table->c node
2.b) Finding all c node’s possible child node.
2.c) Calculate each child node’s f value.
2.d) Remove c node from open table , add it to closed table.
2.e) Add all c node’s child nodes to open table in an undescend sequence.
Wend
3.Ouput Search Result
算法的实现部分参附录1(C Version),附录2(AS2 Version)
提取最优路径的算法并不复杂,虽然在CLOSE表中会有许多无效的搜索点,但是最优路径上各结点的下标一定是按照CLOSE表中下标的升序排列的。因此,只要在CLOSE表中,将下标从终止点向起始点移动,若CLOSE[i+1]与CLOSE[i]没有关联,则剔除CLOSE[i]。
[3A*算法在路径最优问题的交互式示例]
路径最优问题,简单来说,就是在两个结点之间找一条最短路径。有的朋友不禁要问,这个问题不是已经有Dijkstra算法可以解决了吗?此话不假,但是不要忘了Dijkstra算法的复杂度是O(n^2),一旦结点很多并且需要实时计算的话,Dijkstra就无法满足要求了。而A*来处理这类有需要实时要求的问题则显得游刃有余。
在路径最优问题中,用来作为启发函数关键部分的h(n)其实很容易选,那便是当前结点至最终结点的距离,这个距离既可以是Hamilton距离(|x1-x2|+|y1-y2|),亦可以是Euclid 距离(直线距离)。都可以在较快的速度下达到问题的最优解。
下面给出在Flash平台下,用AS2编制的交互式A*算法寻路问题的三个示例程序:(限于Flash的处理性能,当你点了一个目标点后,需要等卡通人物行进到该点时方可点下一点)
http://www.ogdev.net/download/2006/Test1.swf
A* Test1
http://www.ogdev.net/download/2006/Test2.swf
A* Test2
http://www.ogdev.net/download/2006/Test3.swf
A* Test3
[4]小结
本文不仅对A*算法的思想来源及核心部分作了较为详细的介绍,更重要的是,给出了A*算法在网络平台下的一种交互式实现,这一点具有创新意义。
这个程序的实现,不仅给我自己带去了快乐,相信也带给了更多朋友一点小小的启发,对此,我深感颀慰。当然,这个程序还可以在时实交互性上做得更好些,这是值得努力的方向。还有,A*算法本身亦有许多改进之处,如应对“陷井”时,如何更快的脱离,避免无畏的搜索,都是值得去展开的方向。
[参考文献与网站]
[0] 林尧瑞、马少平著,人工智能导论,清华大学出版社,2001.
[1] Nils J.Nilsson著,郑扣根等译,人工智能,机械工业出版社,2003
[2] http://ai.stanford.edu/users/nilsson/trweb/tr.html
[3] http://www.crc.ricoh.com/~hart/
[4] http://i11www.iti.uni-karlsruhe.de/~fschulz/shortest-paths/
附录[0]
A*算法核心部分之C语言版本:
void parseMap_AStar()
{
float tempG,tempF;
EPQ_DataType tempPoint;
int noFindingFlag;
int i;
int moreThanOneBlock;
int sIndex;
currentPoint=startPoint;
//early check
enQueue_EPQ(testQueue,currentPoint,LESS_THAN_VAL);
if(currentPoint->x==targetPoint->x&&
currentPoint->y==targetPoint->y)
goto LABEL_FOUNDED;
//Init for output data
CTCIndex=0;
NAIndex=0;
noFindingFlag=0;
moreThanOneBlock=0;
sIndex=0;
//core algorithm
while(!isEmpty_EPQ(testQueue))
{
currentPoint=deQueue_EPQ(testQueue);
//For natrual A* path finding answer output
CLOSETable[CTCIndex].x=currentPoint->pX;
CLOSETable[CTCIndex].y=currentPoint->pY;
NATable[NAIndex]=CLOSETable[CTCIndex];
CTCIndex++;NAIndex++;
//from current point ,look for the possible point to move.
//start from west ,clockwise
tempG=currentPoint->g+STEP_G;
//west
if(currentPoint->x>1&&
mapData[(int)currentPoint->y][(int)currentPoint->x-1]!=BLOCKED_GROUND&&
mapData[(int)currentPoint->y][(int)currentPoint->x-1]!=LABELED)
{
//target pos check if(currentPoint->x-1==targetPoint->x&&
currentPoint->y==targetPoint->y)
{ currentPoint->pX=currentPoint->x;
currentPoint->pY=currentPoint->y;
currentPoint->x=currentPoint->x-1;
goto LABEL_FOUNDED;
}
tempF=calDis(currentPoint->y,currentPoint->x-1);
tempPoint=createAPoint(currentPoint->x-1,currentPoint->y);
tempPoint->pX=currentPoint->x;
tempPoint->pY=currentPoint->y;
setAStarValue(tempPoint,tempF,tempG);
enQueue_EPQ(testQueue,tempPoint,LESS_THAN_VAL);
mapData[(int)currentPoint->y][(int)currentPoint->x-1]=LABELED;
}
//north
if(currentPoint->y>1&&
mapData[(int)currentPoint->y-1][(int)currentPoint->x]!=BLOCKED_GROUND&&
mapData[(int)currentPoint->y-1][(int)currentPoint->x]!=LABELED)
{
//target pos check if(currentPoint->x==targetPoint->x&&
currentPoint->y-1==targetPoint->y)
{ currentPoint->pX=currentPoint->x;
currentPoint->pY=currentPoint->y;
currentPoint->y=currentPoint->y-1;
goto LABEL_FOUNDED;
}
tempF=calDis(currentPoint->y-1,currentPoint->x);
tempPoint=createAPoint(currentPoint->x,currentPoint->y-1);
tempPoint->pX=currentPoint->x;
tempPoint->pY=currentPoint->y;
setAStarValue(tempPoint,tempF,tempG);
enQueue_EPQ(testQueue,tempPoint,LESS_THAN_VAL);
mapData[(int)currentPoint->y-1][(int)currentPoint->x]=LABELED;
}
//east
if(currentPoint->x
mapData[(int)currentPoint->y][(int)currentPoint->x+1]!=BLOCKED_GROUND&&
mapData[(int)currentPoint->y][(int)currentPoint->x+1]!=LABELED)
{
//target pos check if(currentPoint->x+1==targetPoint->x&&
currentPoint->y==targetPoint->y)
{
currentPoint->pX=currentPoint->x;
currentPoint->pY=currentPoint->y;
currentPoint->x=currentPoint->x+1;
goto LABEL_FOUNDED;
}
tempF=calDis(currentPoint->y,currentPoint->x+1); tempPoint=createAPoint(currentPoint->x+1,currentPoint->y);
tempPoint->pX=currentPoint->x;
tempPoint->pY=currentPoint->y;
setAStarValue(tempPoint,tempF,tempG);
enQueue_EPQ(testQueue,tempPoint,LESS_THAN_VAL);
mapData[(int)currentPoint->y][(int)currentPoint->x+1]=LABELED;
}
//south
if(currentPoint->y
mapData[(int)currentPoint->y+1][(int)currentPoint->x]!=BLOCKED_GROUND&&
mapData[(int)currentPoint->y+1][(int)currentPoint->x]!=LABELED)
{
//target pos check if(currentPoint->x==targetPoint->x&&
currentPoint->y+1==targetPoint->y)
{
currentPoint->pX=currentPoint->x;
currentPoint->pY=currentPoint->y;
currentPoint->y=currentPoint->y+1;
goto LABEL_FOUNDED;
}
tempF=calDis(currentPoint->y+1,currentPoint->x);
tempPoint=createAPoint(currentPoint->x,currentPoint->y+1);
tempPoint->pX=currentPoint->x;
tempPoint->pY=currentPoint->y;
setAStarValue(tempPoint,tempF,tempG);
enQueue_EPQ(testQueue,tempPoint,LESS_THAN_VAL);
mapData[(int)currentPoint->y+1][(int)currentPoint->x]=LABELED;
}
}
LABEL_FOUNDED:
CLOSETable[CTCIndex].x=currentPoint->pX;
CLOSETable[CTCIndex].y=currentPoint->pY;
NATable[NAIndex]=CLOSETable[CTCIndex];
CTCIndex++;NAIndex++;
CLOSETable[CTCIndex].x=currentPoint->x;
CLOSETable[CTCIndex].y=currentPoint->y;
NATable[NAIndex]=CLOSETable[CTCIndex];
CTCIndex++;NAIndex++;
//PROCESS OF ANSTable.
ANSTable[0]=CLOSETable[CTCIndex-1];
ANSIndex=1;
i=CTCIndex-2;
do
{
//ARBITARY OF NEAR BY POINT
if(
(CLOSETable[i].x-1==ANSTable[ANSIndex-1].x&&CLOSETable[i].y==ANSTable[ANSIndex-1].y)||
(CLOSETable[i].x==ANSTable[ANSIndex-1].x&&CLOSETable[i].y-1==ANSTable[ANSIndex-1].y)||
(CLOSETable[i].x+1==ANSTable[ANSIndex-1].x&&CLOSETable[i].y==ANSTable[ANSIndex-1].y)||
(CLOSETable[i].x==ANSTable[ANSIndex-1].x&&CLOSETable[i].y+1==ANSTable[ANSIndex-1].y))
{
ANSTable[ANSIndex]=CLOSETable[i];
ANSIndex++;
}
i--;
}while(!(CLOSETable[i].x==startPoint->x&&CLOSETable[i].y==startPoint->y));
//Output CLOSE table and NATRUAL table … …
}
附录[1]
A*算法核心部分之AS2语言版本:
function AStarSearch():Void
{
if(_root.gStartX==_root.gEndX&&_root.gStartY==_root.gEndY)
return;
else
{
AStarQueue=new eQueue();
targetNode=new eNode();
currentNode=new eNode();
tempNode=new eNode();
//a)data init
NAIndex=0;
CTCIndex=0;
FrameNAIndex=0;
STEP_G=0.5;
noFindingFlag=0;
moreThanOneBlock=0;
sIndex=0;
targetNode.setXY(_root.gEndX,_root.gEndY);
currentNode.setXY(_root.gStartX,_root.gStartY);
_root.gMapArr[_root.gStartY][_root.gStartX]=LABELED;
currentNode.pX=-1;currentNode.pY=-1;
currentNode.setG_H(0,Cal_Cost(currentNode,targetNode));
AStarQueue.enQueue_AStar(currentNode);
//b)core algorithm
while(!AStarQueue.isEmpty())
{
currentNode=AStarQueue.deQueue();
//For natrual A* path finding answer output
CLOSETable[CTCIndex]=new eNode(); CLOSETable[CTCIndex].x=currentNode.pX; CLOSETable[CTCIndex].y=currentNode.pY;
CTCIndex++;
//from current point ,look for the possible point to move.
//start from west ,clockwise
tempG=currentNode.mG+STEP_G;
//west
if(currentNode.x>1&&
_root.gMapArr[currentNode.y][currentNode.x-1]!=BLOCKED_GROUND&&
_root.gMapArr[currentNode.y][currentNode.x-1]!=LABELED)
{
//target pos check
if(currentNode.x-1==targetNode.x&¤tNode.y==targetNode.y)
{
currentNode.pX=currentNode.x;
currentNode.pY=currentNode.y;
currentNode.x=currentNode.x-1;
break;
}
tempNode=new eNode();
tempNode.setXY(currentNode.x-1,currentNode.y);
tempH=Cal_Cost(targetNode,currentNode);
tempNode.pX=currentNode.x;
tempNode.pY=currentNode.y;
tempNode.setG_H(tempG,tempH);
AStarQueue.enQueue_AStar(tempNode);
_root.gMapArr[currentNode.y][currentNode.x-1]=LABELED;
}
//north
if(currentNode.y>1&&
_root.gMapArr[currentNode.y-1][currentNode.x]!=BLOCKED_GROUND&&
_root.gMapArr[currentNode.y-1][currentNode.x]!=LABELED)
{
//target pos check if(currentNode.x==targetNode.x&¤tNode.y-1==targetNode.y)
{
currentNode.pX=currentNode.x;
currentNode.pY=currentNode.y;
currentNode.y=currentNode.y-1;
break;
}
tempNode=new eNode();
tempNode.setXY(currentNode.x,currentNode.y-1);
tempH=Cal_Cost(targetNode,tempNode);
tempNode.pX=currentNode.x;
tempNode.pY=currentNode.y;
tempNode.setG_H(tempG,tempH);
AStarQueue.enQueue_AStar(tempNode);
_root.gMapArr[currentNode.y-1][currentNode.x]=LABELED;
}
//east
if(currentNode.x<_root.gColNum-1&&
_root.gMapArr[currentNode.y][currentNode.x+1]!=BLOCKED_GROUND&&_root.gMapArr[currentNode.y][currentNode.x+1]!=LABELED)
{
//target pos check if(currentNode.x+1==targetNode.x&¤tNode.y==targetNode.y)
{
currentNode.pX=currentNode.x;
currentNode.pY=currentNode.y;
currentNode.x=currentNode.x+1;
break;
}
tempNode=new eNode(); tempNode.setXY(currentNode.x+1,currentNode.y); tempH=Cal_Cost(targetNode,tempNode);
tempNode.pX=currentNode.x;
tempNode.pY=currentNode.y;
tempNode.setG_H(tempG,tempH);
AStarQueue.enQueue_AStar(tempNode);
_root.gMapArr[currentNode.y][currentNode.x+1]=LABELED;
}
//south
if(currentNode.y<_root.gRowNum-1&&
_root.gMapArr[currentNode.y+1][currentNode.x]!=BLOCKED_GROUND&&
_root.gMapArr[currentNode.y+1][currentNode.x]!=LABELED)
{
//target pos check if(currentNode.x==targetNode.x&¤tNode.y+1==targetNode.y)
{
currentNode.pX=currentNode.x;
currentNode.pY=currentNode.y;
currentNode.y=currentNode.y+1;
break;
}
tempNode=new eNode();
tempNode.setXY(currentNode.x,currentNode.y+1);
tempH=Cal_Cost(tempNode,targetNode);
tempNode.pX=currentNode.x;
tempNode.pY=currentNode.y;
tempNode.setG_H(tempG,tempH);
AStarQueue.enQueue_AStar(tempNode);
_root.gMapArr[currentNode.y+1][currentNode.x]=LABELED;
}
}
//FINISHED_LABEL:
CLOSETable[CTCIndex]=new eNode();
CLOSETable[CTCIndex].x=currentNode.pX;
CLOSETable[CTCIndex].y=currentNode.pY;
CTCIndex++;
CLOSETable[CTCIndex]=new eNode();
CLOSETable[CTCIndex].x=currentNode.x;
CLOSETable[CTCIndex].y=currentNode.y;
CTCIndex++;
//PROCESS OF NATable.
NATable[0]=new eNode();
NATable[0]=CLOSETable[CTCIndex-1];
NAIndex=1;
i=CTCIndex-2;
do
{
//ARBITARY OF NEAR BY POINT
if(
(CLOSETable[i].x-1==NATable[NAIndex-1].x&&CLOSETable[i].y==NATable[NAIndex-1].y)||
(CLOSETable[i].x==NATable[NAIndex-1].x&&CLOSETable[i].y-1==NATable[NAIndex-1].y)||
(CLOSETable[i].x+1==NATable[NAIndex-1].x&&CLOSETable[i].y==NATable[NAIndex-1].y)||
(CLOSETable[i].x==NATable[NAIndex-1].x&&CLOSETable[i].y+1==NATable[NAIndex-1].y))
{
NATable[NAIndex]=new eNode();
NATable[NAIndex]=CLOSETable[i];
NAIndex++;
}
i--;
if(i==0)//to process only 1 step move
break;
}while(true);//Important revise ,as some child node will adjust the root node.
var timeCount:Number=0;
var midObj:eNode;
if(NAIndex%2==0)
timeCount=NAIndex/2;
else
timeCount=(NAIndex-1)/2;
for(i=0;i
{
midObj=new eNode();
midObj=NATable[i];
NATable[i]=NATable[NAIndex-1-i];
NATable[NAIndex-1-i]=midObj;
}
}
}
[源码下载、源程序下载]
源程序内容列表:
Code1:C语言版本源码
Code2:AS2版本源码
Code3:三个不同地图的效果展示
http://www.ogdev.net/download/2006/code1.rar
http://www.ogdev.net/download/2006/code2.rar
http://www.ogdev.net/download/2006/code3.rar
A* Algorithm Principle and Practice
[Classify] Algorithm Implementation , Artificial Intelligence
[ Level ] ★★★★★
[Abstract] This article introduces an important and effective heuristics algorithm ------ A* algorithm with its theory and also gives out its interactive implementation of path finding problem.
[Key Words] A* , Heuristics Algorithm , Best Path Finding , Interactive , AS2
[1历史回顾]
P. E. Hart , N. J. Nilsson 和B. Raphael共同发表了一篇在启发式搜索方面有深远影响力的论文:
“P. E. Hart, N. J. Nilsson, and B. Raphael. A formal basis for the heuristic determination of minimum cost paths in graphs. IEEE Trans. Syst. Sci. and Cybernetics, SSC-4(2):100-107, 1968”。从此,一种精巧、高效的算法------A*算法横空出世了,并在相关领域得到了广泛的应用。
Peter E. Hart
Nils J. Nilsson
[2从DFS和BFS中来]
A*算法的思想来源并不是什么高深莫测的东西,事实上,它与我们所熟悉的另外两种搜索策略:DFS(深度优先Deep First Search)和 BFS(广度优先Breadth First Search)有着自然而紧密的联系。
首先要提一下搜索树的概念,一个可以搜索出某个可行解的问题,如“农夫、白菜、羊、狼”和“八皇后”等,虽然从表面上看上去和“树”这种结构无关,但是整个搜索过程中的可能试探点所行成的搜索空间总可以对应到一颗搜索树上去。所以,将各类形式上不同的搜索问题抽象并统一成为搜索树的形式,为算法的设计与分析带来巨大的方便。
下面,让我们通过两个Flash演示来回顾一下DFS和BFS的算法思想:
DFS算法思想演示:
http://www.ogdev.net/download/2006/deep.swf
BFS算法思想演示:
http://www.ogdev.net/download/2006/breadth.swf
DFS和BFS在展开子结点时均属于盲目型搜索,也就是说,它不会选择哪个结点在下一次搜索中更优而去跳转到该结点进行下一步的搜索。在运气不好的情形中,均需要试探完整个解集空间, 显然,只能适用于问题规模不大的搜索问题中。
那么,作为启发式算法中的A*算法,又比它们高效在哪里呢?
首先要来谈一下什么是启发式算法。所谓启发式搜索,与DFS和BFS这类盲目型搜索最大的不同,就在于当前搜索结点往下选择下一步结点时,可以通过一个启发函数来进行选择,选择代价最少的结点作为下一步搜索结点而跳转其上(遇到有一个以上代价最少的结点,不妨选距离当前搜索点最近一次展开的搜索点进行下一步搜索)。一个经过仔细设计的启发函数,往往在很快的时间内就可得到一个搜索问题的最优解,对于NP问题,亦可在多项式时间内得到一个较优解。
A*算法,作为启发式算法中很重要的一种,被广泛应用在最优路径求解和一些策略设计的问题中。而A*算法最为核心的部分,就在于它的一个估值函数的设计上:
f(n)=g(n)+h(n)
其中f(n)是每个可能试探点的估值,它有两部分组成:一部分为g(n),它表示从起始搜索点到当前点的代价(通常用某结点在搜索树中的深度来表示)。另一部分,即h(n),它表示启发式搜索中最为重要的一部分,即当前结点到目标结点的估值,h(n)设计的好坏,直接影响着具有此种启发式函数的启发式算法的是否能称为A*算法。
一种具有f(n)=g(n)+h(n)策略的启发式算法能成为A*算法的充分条件是:
1) 搜索树上存在着从起始点到终了点的最优路径。
2) 问题域是有限的。
3)所有结点的子结点的搜索代价值>0。
4)h(n)=
当此四个条件都满足时,一个具有f(n)=g(n)+h(n)策略的启发式算法能成为A*算法,并一定能找到最优解。([1]P89给出了相关的证明)
对于一个搜索问题,显然,条件1,2,3都是很容易满足的,而
条件4): h(n)<=h*(n)是需要精心设计的,由于h*(n)显然是无法知道的,
所以,一个满足条件4)的启发策略h(n)就来的难能可贵了。不过,对于图的最优路径搜索和八数码问题,有些相关策略h(n)不仅很好理解,而且已经在理论上证明是满足条件4)的,从而为这个算法的推广起到了决定性的作用。不过h(n)距离h*(n)的呈度不能过大,否则h(n)就没有过强的区分能力,算法效率并不会很高。对一个好的h(n)的评价是:h(n)在h*(n)的下界之下,并且尽量接近h*(n).
当然,估值函数的设计也就就仅仅是f(n)=g(n)+h(n)一种,另外的估值函数“变种”如:f(n)=w*g(n)+(1-w)*h(n) ,f(n)=g(n)+h(n)+h(n-1)针对不同的具体问题亦会有不同的效果。
[3A*算法的核心过程]
让我们先来通过两个Flash演示,来看一下A*算法在具体应用时的搜索树是如何展开的。
http://www.ogdev.net/download/2006/searchsample1.swf
A* Search搜索树1
http://www.ogdev.net/download/2006/searchsample1.swf
A* Search搜索树2
通过演示,我们可以看出:A*算法最为核心的过程,就在每次选择下一个当前搜索点时,是从所有已探知的但未搜索过点中(可能是不同层,亦可不在同一条支路上),选取f值最小的结点进行展开。而所有“已探知的但未搜索过点”可以通过一个按f值升序的队列(即优先队列)进行排列。这样,在整体的搜索过程中,只要按照类似广度优先的算法框架,从优先队列中弹出队首元素(f值),对其可能子结点计算g、h和f值,直到优先队列为空(无解)或找到终止点为止。
A*算法与广度优先和深度优先的联系就在于,当g(n)=0时,该算法类似于DFS,当h(n)=0时,该算法类似于BFS , 这一点,可以通过上面的A*搜索树的具体过程中将h(n)设为0或将g(n)设为0而得到。
A*算法实现框架:
重要数据解释:
Open Table :存放所有已探知的但未搜索过点的优先队列。
Closed Table :存在搜索过的点的数组,提取最优路径时有用。
Start Node :起始点。
Target Node :终止点。
C Node :当前点。
算法框架如下:
1.Init start node , add it to open table
While not reach target node && open table is unNull
2.a) Get the head node in open table->c node
2.b) Finding all c node’s possible child node.
2.c) Calculate each child node’s f value.
2.d) Remove c node from open table , add it to closed table.
2.e) Add all c node’s child nodes to open table in an undescend sequence.
Wend
3.Ouput Search Result
算法的实现部分参附录1(C Version),附录2(AS2 Version)
提取最优路径的算法并不复杂,虽然在CLOSE表中会有许多无效的搜索点,但是最优路径上各结点的下标一定是按照CLOSE表中下标的升序排列的。因此,只要在CLOSE表中,将下标从终止点向起始点移动,若CLOSE[i+1]与CLOSE[i]没有关联,则剔除CLOSE[i]。
[3A*算法在路径最优问题的交互式示例]
路径最优问题,简单来说,就是在两个结点之间找一条最短路径。有的朋友不禁要问,这个问题不是已经有Dijkstra算法可以解决了吗?此话不假,但是不要忘了Dijkstra算法的复杂度是O(n^2),一旦结点很多并且需要实时计算的话,Dijkstra就无法满足要求了。而A*来处理这类有需要实时要求的问题则显得游刃有余。
在路径最优问题中,用来作为启发函数关键部分的h(n)其实很容易选,那便是当前结点至最终结点的距离,这个距离既可以是Hamilton距离(|x1-x2|+|y1-y2|),亦可以是Euclid 距离(直线距离)。都可以在较快的速度下达到问题的最优解。
下面给出在Flash平台下,用AS2编制的交互式A*算法寻路问题的三个示例程序:(限于Flash的处理性能,当你点了一个目标点后,需要等卡通人物行进到该点时方可点下一点)
http://www.ogdev.net/download/2006/Test1.swf
A* Test1
http://www.ogdev.net/download/2006/Test2.swf
A* Test2
http://www.ogdev.net/download/2006/Test3.swf
A* Test3
[4]小结
本文不仅对A*算法的思想来源及核心部分作了较为详细的介绍,更重要的是,给出了A*算法在网络平台下的一种交互式实现,这一点具有创新意义。
这个程序的实现,不仅给我自己带去了快乐,相信也带给了更多朋友一点小小的启发,对此,我深感颀慰。当然,这个程序还可以在时实交互性上做得更好些,这是值得努力的方向。还有,A*算法本身亦有许多改进之处,如应对“陷井”时,如何更快的脱离,避免无畏的搜索,都是值得去展开的方向。
[参考文献与网站]
[0] 林尧瑞、马少平著,人工智能导论,清华大学出版社,2001.
[1] Nils J.Nilsson著,郑扣根等译,人工智能,机械工业出版社,2003
[2] http://ai.stanford.edu/users/nilsson/trweb/tr.html
[3] http://www.crc.ricoh.com/~hart/
[4] http://i11www.iti.uni-karlsruhe.de/~fschulz/shortest-paths/
附录[0]
A*算法核心部分之C语言版本:
void parseMap_AStar()
{
float tempG,tempF;
EPQ_DataType tempPoint;
int noFindingFlag;
int i;
int moreThanOneBlock;
int sIndex;
currentPoint=startPoint;
//early check
enQueue_EPQ(testQueue,currentPoint,LESS_THAN_VAL);
if(currentPoint->x==targetPoint->x&&
currentPoint->y==targetPoint->y)
goto LABEL_FOUNDED;
//Init for output data
CTCIndex=0;
NAIndex=0;
noFindingFlag=0;
moreThanOneBlock=0;
sIndex=0;
//core algorithm
while(!isEmpty_EPQ(testQueue))
{
currentPoint=deQueue_EPQ(testQueue);
//For natrual A* path finding answer output
CLOSETable[CTCIndex].x=currentPoint->pX;
CLOSETable[CTCIndex].y=currentPoint->pY;
NATable[NAIndex]=CLOSETable[CTCIndex];
CTCIndex++;NAIndex++;
//from current point ,look for the possible point to move.
//start from west ,clockwise
tempG=currentPoint->g+STEP_G;
//west
if(currentPoint->x>1&&
mapData[(int)currentPoint->y][(int)currentPoint->x-1]!=BLOCKED_GROUND&&
mapData[(int)currentPoint->y][(int)currentPoint->x-1]!=LABELED)
{
//target pos check if(currentPoint->x-1==targetPoint->x&&
currentPoint->y==targetPoint->y)
{ currentPoint->pX=currentPoint->x;
currentPoint->pY=currentPoint->y;
currentPoint->x=currentPoint->x-1;
goto LABEL_FOUNDED;
}
tempF=calDis(currentPoint->y,currentPoint->x-1);
tempPoint=createAPoint(currentPoint->x-1,currentPoint->y);
tempPoint->pX=currentPoint->x;
tempPoint->pY=currentPoint->y;
setAStarValue(tempPoint,tempF,tempG);
enQueue_EPQ(testQueue,tempPoint,LESS_THAN_VAL);
mapData[(int)currentPoint->y][(int)currentPoint->x-1]=LABELED;
}
//north
if(currentPoint->y>1&&
mapData[(int)currentPoint->y-1][(int)currentPoint->x]!=BLOCKED_GROUND&&
mapData[(int)currentPoint->y-1][(int)currentPoint->x]!=LABELED)
{
//target pos check if(currentPoint->x==targetPoint->x&&
currentPoint->y-1==targetPoint->y)
{ currentPoint->pX=currentPoint->x;
currentPoint->pY=currentPoint->y;
currentPoint->y=currentPoint->y-1;
goto LABEL_FOUNDED;
}
tempF=calDis(currentPoint->y-1,currentPoint->x);
tempPoint=createAPoint(currentPoint->x,currentPoint->y-1);
tempPoint->pX=currentPoint->x;
tempPoint->pY=currentPoint->y;
setAStarValue(tempPoint,tempF,tempG);
enQueue_EPQ(testQueue,tempPoint,LESS_THAN_VAL);
mapData[(int)currentPoint->y-1][(int)currentPoint->x]=LABELED;
}
//east
if(currentPoint->x
mapData[(int)currentPoint->y][(int)currentPoint->x+1]!=BLOCKED_GROUND&&
mapData[(int)currentPoint->y][(int)currentPoint->x+1]!=LABELED)
{
//target pos check if(currentPoint->x+1==targetPoint->x&&
currentPoint->y==targetPoint->y)
{
currentPoint->pX=currentPoint->x;
currentPoint->pY=currentPoint->y;
currentPoint->x=currentPoint->x+1;
goto LABEL_FOUNDED;
}
tempF=calDis(currentPoint->y,currentPoint->x+1); tempPoint=createAPoint(currentPoint->x+1,currentPoint->y);
tempPoint->pX=currentPoint->x;
tempPoint->pY=currentPoint->y;
setAStarValue(tempPoint,tempF,tempG);
enQueue_EPQ(testQueue,tempPoint,LESS_THAN_VAL);
mapData[(int)currentPoint->y][(int)currentPoint->x+1]=LABELED;
}
//south
if(currentPoint->y
mapData[(int)currentPoint->y+1][(int)currentPoint->x]!=BLOCKED_GROUND&&
mapData[(int)currentPoint->y+1][(int)currentPoint->x]!=LABELED)
{
//target pos check if(currentPoint->x==targetPoint->x&&
currentPoint->y+1==targetPoint->y)
{
currentPoint->pX=currentPoint->x;
currentPoint->pY=currentPoint->y;
currentPoint->y=currentPoint->y+1;
goto LABEL_FOUNDED;
}
tempF=calDis(currentPoint->y+1,currentPoint->x);
tempPoint=createAPoint(currentPoint->x,currentPoint->y+1);
tempPoint->pX=currentPoint->x;
tempPoint->pY=currentPoint->y;
setAStarValue(tempPoint,tempF,tempG);
enQueue_EPQ(testQueue,tempPoint,LESS_THAN_VAL);
mapData[(int)currentPoint->y+1][(int)currentPoint->x]=LABELED;
}
}
LABEL_FOUNDED:
CLOSETable[CTCIndex].x=currentPoint->pX;
CLOSETable[CTCIndex].y=currentPoint->pY;
NATable[NAIndex]=CLOSETable[CTCIndex];
CTCIndex++;NAIndex++;
CLOSETable[CTCIndex].x=currentPoint->x;
CLOSETable[CTCIndex].y=currentPoint->y;
NATable[NAIndex]=CLOSETable[CTCIndex];
CTCIndex++;NAIndex++;
//PROCESS OF ANSTable.
ANSTable[0]=CLOSETable[CTCIndex-1];
ANSIndex=1;
i=CTCIndex-2;
do
{
//ARBITARY OF NEAR BY POINT
if(
(CLOSETable[i].x-1==ANSTable[ANSIndex-1].x&&CLOSETable[i].y==ANSTable[ANSIndex-1].y)||
(CLOSETable[i].x==ANSTable[ANSIndex-1].x&&CLOSETable[i].y-1==ANSTable[ANSIndex-1].y)||
(CLOSETable[i].x+1==ANSTable[ANSIndex-1].x&&CLOSETable[i].y==ANSTable[ANSIndex-1].y)||
(CLOSETable[i].x==ANSTable[ANSIndex-1].x&&CLOSETable[i].y+1==ANSTable[ANSIndex-1].y))
{
ANSTable[ANSIndex]=CLOSETable[i];
ANSIndex++;
}
i--;
}while(!(CLOSETable[i].x==startPoint->x&&CLOSETable[i].y==startPoint->y));
//Output CLOSE table and NATRUAL table … …
}
附录[1]
A*算法核心部分之AS2语言版本:
function AStarSearch():Void
{
if(_root.gStartX==_root.gEndX&&_root.gStartY==_root.gEndY)
return;
else
{
AStarQueue=new eQueue();
targetNode=new eNode();
currentNode=new eNode();
tempNode=new eNode();
//a)data init
NAIndex=0;
CTCIndex=0;
FrameNAIndex=0;
STEP_G=0.5;
noFindingFlag=0;
moreThanOneBlock=0;
sIndex=0;
targetNode.setXY(_root.gEndX,_root.gEndY);
currentNode.setXY(_root.gStartX,_root.gStartY);
_root.gMapArr[_root.gStartY][_root.gStartX]=LABELED;
currentNode.pX=-1;currentNode.pY=-1;
currentNode.setG_H(0,Cal_Cost(currentNode,targetNode));
AStarQueue.enQueue_AStar(currentNode);
//b)core algorithm
while(!AStarQueue.isEmpty())
{
currentNode=AStarQueue.deQueue();
//For natrual A* path finding answer output
CLOSETable[CTCIndex]=new eNode(); CLOSETable[CTCIndex].x=currentNode.pX; CLOSETable[CTCIndex].y=currentNode.pY;
CTCIndex++;
//from current point ,look for the possible point to move.
//start from west ,clockwise
tempG=currentNode.mG+STEP_G;
//west
if(currentNode.x>1&&
_root.gMapArr[currentNode.y][currentNode.x-1]!=BLOCKED_GROUND&&
_root.gMapArr[currentNode.y][currentNode.x-1]!=LABELED)
{
//target pos check
if(currentNode.x-1==targetNode.x&¤tNode.y==targetNode.y)
{
currentNode.pX=currentNode.x;
currentNode.pY=currentNode.y;
currentNode.x=currentNode.x-1;
break;
}
tempNode=new eNode();
tempNode.setXY(currentNode.x-1,currentNode.y);
tempH=Cal_Cost(targetNode,currentNode);
tempNode.pX=currentNode.x;
tempNode.pY=currentNode.y;
tempNode.setG_H(tempG,tempH);
AStarQueue.enQueue_AStar(tempNode);
_root.gMapArr[currentNode.y][currentNode.x-1]=LABELED;
}
//north
if(currentNode.y>1&&
_root.gMapArr[currentNode.y-1][currentNode.x]!=BLOCKED_GROUND&&
_root.gMapArr[currentNode.y-1][currentNode.x]!=LABELED)
{
//target pos check if(currentNode.x==targetNode.x&¤tNode.y-1==targetNode.y)
{
currentNode.pX=currentNode.x;
currentNode.pY=currentNode.y;
currentNode.y=currentNode.y-1;
break;
}
tempNode=new eNode();
tempNode.setXY(currentNode.x,currentNode.y-1);
tempH=Cal_Cost(targetNode,tempNode);
tempNode.pX=currentNode.x;
tempNode.pY=currentNode.y;
tempNode.setG_H(tempG,tempH);
AStarQueue.enQueue_AStar(tempNode);
_root.gMapArr[currentNode.y-1][currentNode.x]=LABELED;
}
//east
if(currentNode.x<_root.gColNum-1&&
_root.gMapArr[currentNode.y][currentNode.x+1]!=BLOCKED_GROUND&&_root.gMapArr[currentNode.y][currentNode.x+1]!=LABELED)
{
//target pos check if(currentNode.x+1==targetNode.x&¤tNode.y==targetNode.y)
{
currentNode.pX=currentNode.x;
currentNode.pY=currentNode.y;
currentNode.x=currentNode.x+1;
break;
}
tempNode=new eNode(); tempNode.setXY(currentNode.x+1,currentNode.y); tempH=Cal_Cost(targetNode,tempNode);
tempNode.pX=currentNode.x;
tempNode.pY=currentNode.y;
tempNode.setG_H(tempG,tempH);
AStarQueue.enQueue_AStar(tempNode);
_root.gMapArr[currentNode.y][currentNode.x+1]=LABELED;
}
//south
if(currentNode.y<_root.gRowNum-1&&
_root.gMapArr[currentNode.y+1][currentNode.x]!=BLOCKED_GROUND&&
_root.gMapArr[currentNode.y+1][currentNode.x]!=LABELED)
{
//target pos check if(currentNode.x==targetNode.x&¤tNode.y+1==targetNode.y)
{
currentNode.pX=currentNode.x;
currentNode.pY=currentNode.y;
currentNode.y=currentNode.y+1;
break;
}
tempNode=new eNode();
tempNode.setXY(currentNode.x,currentNode.y+1);
tempH=Cal_Cost(tempNode,targetNode);
tempNode.pX=currentNode.x;
tempNode.pY=currentNode.y;
tempNode.setG_H(tempG,tempH);
AStarQueue.enQueue_AStar(tempNode);
_root.gMapArr[currentNode.y+1][currentNode.x]=LABELED;
}
}
//FINISHED_LABEL:
CLOSETable[CTCIndex]=new eNode();
CLOSETable[CTCIndex].x=currentNode.pX;
CLOSETable[CTCIndex].y=currentNode.pY;
CTCIndex++;
CLOSETable[CTCIndex]=new eNode();
CLOSETable[CTCIndex].x=currentNode.x;
CLOSETable[CTCIndex].y=currentNode.y;
CTCIndex++;
//PROCESS OF NATable.
NATable[0]=new eNode();
NATable[0]=CLOSETable[CTCIndex-1];
NAIndex=1;
i=CTCIndex-2;
do
{
//ARBITARY OF NEAR BY POINT
if(
(CLOSETable[i].x-1==NATable[NAIndex-1].x&&CLOSETable[i].y==NATable[NAIndex-1].y)||
(CLOSETable[i].x==NATable[NAIndex-1].x&&CLOSETable[i].y-1==NATable[NAIndex-1].y)||
(CLOSETable[i].x+1==NATable[NAIndex-1].x&&CLOSETable[i].y==NATable[NAIndex-1].y)||
(CLOSETable[i].x==NATable[NAIndex-1].x&&CLOSETable[i].y+1==NATable[NAIndex-1].y))
{
NATable[NAIndex]=new eNode();
NATable[NAIndex]=CLOSETable[i];
NAIndex++;
}
i--;
if(i==0)//to process only 1 step move
break;
}while(true);//Important revise ,as some child node will adjust the root node.
var timeCount:Number=0;
var midObj:eNode;
if(NAIndex%2==0)
timeCount=NAIndex/2;
else
timeCount=(NAIndex-1)/2;
for(i=0;i
{
midObj=new eNode();
midObj=NATable[i];
NATable[i]=NATable[NAIndex-1-i];
NATable[NAIndex-1-i]=midObj;
}
}
}
[源码下载、源程序下载]
源程序内容列表:
Code1:C语言版本源码
Code2:AS2版本源码
Code3:三个不同地图的效果展示
http://www.ogdev.net/download/2006/code1.rar
http://www.ogdev.net/download/2006/code2.rar
http://www.ogdev.net/download/2006/code3.rar
发表评论
-
水果忍者鼠标跟随特效制作[转载]
2012-03-01 16:06 2457实现这效果其实比较简单,主要是思路~! package ... -
[转]三次贝尔曲线
2011-11-10 01:09 1932http://bbs.9ria.com/viewt ... -
轻量级Eval库Grak轻量级Eval库Grak
2011-09-22 03:07 0轻量级Eval库Grak -
[转]AS3与数据结构
2011-09-14 01:08 0http://www.nshen.net/dataSt ... -
井字棋算法
2011-08-18 15:04 0井字棋算法井字棋算法 -
[转 自己改的一个滚动条类,理论上什么都能“滚”
2011-08-11 23:14 0http://bbs.9ria.com/viewthread. ... -
[转] 关于一段时间内鼠标没有移动,执行某函数
2011-08-10 00:22 0http://bbs.9ria.com/viewthread. ... -
很好的FLEX表情聊天界面
2011-08-09 02:06 0很好的FLEX表情聊天界面 -
Flash版《植物大战僵尸》源码
2011-08-09 01:34 0本帖最后由 IJUST 于 2 ... -
愤怒的小鸟 BOX2D FLASH
2011-08-09 01:27 0姊妹篇:Flash版《植物大战僵尸》源码今年就要大四啦,放暑假 ... -
[转]如何计算线段和圆的交点
2011-08-09 00:53 0http://www.thecodeway.com/b ... -
[转] 45度地图坐标转换
2011-07-30 02:41 0昨天有朋友问我 45度地图中关于鼠标点击如果进行坐标转化 ... -
[转]一个Collision类,其中的block方法可以实现两个物体之间的碰撞检测。
2011-07-30 02:35 1402第二个是书中的源代码给出了一个Collision类,其中 ... -
AS3的一些优化计算方法
2011-07-06 12:56 0http://www.cnitblog.com/flashli ... -
[转]AS3类:CRC32校验类
2011-07-06 12:54 2611http://www.cnitblog.com/flashli ... -
基于哈希表数据源的A星寻路算法 - [as 3.0]
2011-06-16 17:03 0在这贴的代码是为了有需要的人学习而不是 提供源码给别人用的 ... -
计算几何算法概览
2011-06-14 17:28 2124计算几何算法概览 一、引言 ... -
[演示] 判断点是否处于三角形内的算法分析
2011-06-14 17:26 3313http://bbs.wow8.org/thread-9429 ... -
判断点在直线的哪一侧
2011-06-14 17:04 0判断点在直线的哪一侧 2.2.1下面开始程序的设计: ... -
[转]动画中坐标旋转公式的原理
2011-05-25 23:30 1500有一定的其它语言编程基础,所以学习新语言还是比较快的。正在进军 ...
相关推荐
算法引论-一种创造性方法 完整书签pdf版
算法引论--一种创造性方法,里面很多实用算法,和考试有关
计算机算法引论——设计与分析技术.本书讲述计算机算法的各种设计策略,包括分治技术、贪心技术、动态规划技术回溯和分支限界技术等。介绍算法分析技术、算法的时间和空间复杂度分析方法;讨论各类经典的应用问题...
- **目标**:通过本课程的学习,使学生掌握算法设计与分析的基本能力,能够独立完成简单的算法设计任务。 - **背景知识**:需要具备一定的程序设计语言基础、数据结构知识、高等数学、离散数学及概率论的基础。 ###...
Udi Manber编著,黄林鹏译,Introduction to Algorithms:A creative approach
《算法引论》是一本深入探讨算法理论与实践的书籍,它涵盖了计算机科学中的核心算法,为学习者提供了丰富的知识和理解。算法是计算机科学的灵魂,是解决复杂问题的关键工具,而《算法引论》正是引导读者步入这个领域...
### 算法设计与分析期末复习知识点及习题详解 #### 第1章 算法引论 **1.1 算法与程序** - **算法的定义**: 算法是一种精确且完整的解题方案描述,它是解决问题的具体方法和步骤。 - **算法的特征**: - **输入**...
《计算机算法引论:设计与分析技术》是一本面向计算机、软件工程和网络工程专业及相关专业的本科生(高年级)和研究生教材,根据国内外计算机技术的最新发展,讲述计算机算法的各种设计策略,包括分治技术、贪心技术、...
### 编译原理第一章引论知识点详解 #### 一、编译器概念及工作原理 **1.1 编译器定义** - **定义**: 编译器是一种软件工具,其主要职责是将源代码(一种高级编程语言)转换为目标代码(通常是机器语言或另一种...
根据提供的标题“计算机科学引论重点知识及课后答案”以及描述“计算机科学引论重点知识及课后答案,里面包含习题讲解,课后专业单词翻译非常全”,我们可以推测这份资料主要涵盖了计算机科学基础理论的关键知识点,...
算法引论:一种创造性方法 中文版 电子工业出版社 独具特色地将数学归纳法作为讲解和设计算法的工具
### 山东大学计算引论2017-2018期末考试知识点解析 #### 一、简答题解析 **1. DFA与图灵机的区别** - **定义:** - **确定有限自动机(DFA)**:是一种简单的模型,用于识别正则语言。它具有有限的状态集合,每...
这些知识点构成了《算法引论》的主要内容,通过深入学习和实践,读者可以建立起坚实的算法基础,为后续的计算机科学学习和工作打下坚实的基础。在实际应用中,结合具体场景灵活运用这些算法,能够有效地提升问题解决...
4. **排序与搜索算法**:排序(如冒泡排序、快速排序、归并排序等)和搜索(如线性搜索、二分搜索等)是最常见的算法,书中会有详尽的讲解,包括算法的实现、时间复杂度分析和适用场景。 5. **图论算法**:书中可能...
#### 附录A:C++类库管理系统的分析与设计 - **面向对象分析** - 包括需求分析、建立对象模型等步骤。 - **面向对象设计** - 设计类库结构、问题域子系统等。 #### 附录B:一个汉字行编辑程序的设计 - **设计...
总结来说,算法引论及简单算法的学习涵盖了算法设计的基础,特别是算法分析的关键概念,包括时间复杂性和空间复杂性,以及如何通过理论和实践相结合的方式来评估和优化算法。这些知识对于理解和编写高效代码至关重要...
蒙特卡罗方法引,蒙特卡洛算法 蒙特卡罗方法引论--入门和深入分析
《算法引论——一种创造性方法》是一本深入探讨算法设计与分析的经典著作。该书旨在引导读者理解算法的本质,培养解决复杂问题的创造性思维。在IT行业中,算法是解决问题、优化程序性能的关键工具,尤其在大数据处理...