分享
 
 
 

A*算法实现8数或者15数问题(C#实现)

王朝c#·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

8数和15数问题

-、问题描述

8数或15数问题是同一个问题,其就是一个随机排列的8个或15个数在一个方正格子中移动到达规定的一个目标状态。由于只有一个空格子,故只有在空格附近的棋子可以移动。

二、算法描述

F 算法选择

此问题需要对所有可能的路径进行穷举,但是由于随着树的高度的加大,其子结点的增加宿舍剧增,所以对所有的子结点进行穷举是不大现实的。而根据当前的状态和目标状态进行对比可以用一个评估函数来评估当前状态的好坏情况。而在这里我们选择A*算法来进行求解,A*算法是一种最好优先的算法。f'(n) = g'(n) + h'(n),f'(n)是估价函数,g'(n)是起点到终点的最短路径值,这里就表示树的高度,h'(n)是n到目标的最断路经的启发值,其原理是把当前产生的状态和以前所以的状态的评估函数值进行对比,选择其中最小的评估函数继续进行下一步。这里我选择h'(n)的启发值为每个格子到达它的目标位置所需要经过的最少步数。

F 算法描述

需要说明的几点:

1. 用OpenArr表示初始节点列表(待扩展,此为一个动态数组)

2.ClosedArr保存已经访问过的结点

3.算法首先需要给出初始状态,由于随机产生的状态并不一定能够走到目标状态,因此这里采用从标准状态往回走产生一个被打乱的随机状态,这样可以保证有解。

算法实现:

1. OpenArr放置初始结点

2. 如果OpenArr为空集,则退出并给出失败信号

3. n取为OpenArr的第一个节点,并从OpenArr中删除节点n

4. 如果n为目标节点,则退出并给出成功信号

5. 否则,将产生n子节点,并对n的每个子结点n’ 进行如下操作:

1)如果n’ 不在OpenArr和ClosedArr表中,则把n’放入OpenArr表中

2)否则,如果在OpenArr表中,计算评估值,并且如果比表中的评估函数的值小,则更新表中的评估值为当前的值。

3)否则,如果在ClosedArr表中,计算评估值,并且如果比表中的评估函数的值小,则把表中的评估值更新为当前的值,并把该结点从表中删除,并在OpenArr表中加入该结点。

6、把n结点加入ClosedArr中

7、对OpenArr进行排序(根据评估函数从小到大),并转到2。

三、程序设计

算法使用C#语言来实现的。程序主要根据上面提供的广度优先算法的描述来对算法进行实现的。程序共有四个类:

StepNode类,用来描述产生的各个结点的属性以及方法等

Heuristic_15Num_Search类,算法实现类

Form1类,是界面设计的类。

这里分别提供了8数和15数问题的求解。并显示了所经历过的各个状态转移

以下主要对几个核心算法的程序实现进行说明介绍。

//StepNode类 以下几个方法主要是控制格子的左右上下的移动

//0 数字上移

private void MoveUp(Position p)

{

if(p.x>=1)

{

StepNode node = new StepNode();

To(this,node);

Position p1 = new Position(p.x-1,p.y);

AddChild(node,p,p1);

}

}

// 0 数字下移

private void MoveDown(Position p)

{

if(p.x<=text_v.Length-2)

{

StepNode node = new StepNode();

To(this,node);

Position p1 = new Position(p.x+1,p.y);

AddChild(node,p,p1);

}

}

//0 数字左移

private void MoveLeft(Position p)

{

if(p.y>=1)

{

StepNode node = new StepNode();

To(this,node);

Position p1 = new Position(p.x,p.y-1);

AddChild(node,p,p1);

}

}

//0 数字右移

private void MoveRight(Position p)

{

if(p.y<=text_v.Length-2)

{

StepNode node = new StepNode();

To(this,node);

Position p1 = new Position(p.x,p.y+1);

AddChild(node,p,p1);

}

}

//计算节点的启发函数的值

private void ComputeGeuristicGeneVal()

{

int geneVal = this.Height ;

int g = 0; //启发因子 每个数据到达标准位置需要走过的最少步数

for(int i=0;i<text_v.Length;i++)

{

for(int j=0;j<text_v[0].Length;j++)

{

Position p1 = GetPosition(text_v[i][j]);

Position p2 = GetPosition(aim[i,j]);

int xd = p1.x > p2.x ? p1.x - p2.x : p2.x - p1.x ;

int yd = p1.y > p2.y ? p1.y - p2.y : p2.y - p1.y ;

g += xd + yd ;

}

}

geneVal += g;

this.Heuristic_Gene = geneVal;

}

//Heuristic_15Num_Search类

//核心算法实现部分

//循环搜索匹配

private void LoopSearch(StepNode node)

{

while(OpenArr.Count>0)

{

node = (StepNode)OpenArr[0];

OpenArr.Remove(node);

//如果是目标节点则停止搜索

if(node.IsAim())

{

SetPath(node);

return;

}

else

{

//产生子节点。

node.CreateChildren();

//对每个子节点进行检查

foreach(StepNode i in node.Children)

{

//如果不在open和close表中。

if( Contain(ClosedArr,i)==-1 && Contain(OpenArr,i)==-1 )

{

OpenArr.Add(i);

}

//如果在open表中

else if(Contain(OpenArr,i)!=-1)

{

StepNode n = (StepNode)OpenArr[Contain(OpenArr,i)];

//如果i的估计值小于open表中的估计值则替换表中的估计值

if(i.Heuristic_Gene<n.Heuristic_Gene)

{

n.Heuristic_Gene = i.Heuristic_Gene;

}

}

//如果在close中。

else

{

StepNode n = (StepNode)ClosedArr[Contain(ClosedArr,i)];

//如果i的估计值小于closed表中的估计值则替换表中的估计值

if(i.Heuristic_Gene<n.Heuristic_Gene)

{

n.Heuristic_Gene = i.Heuristic_Gene;

ClosedArr.RemoveAt(Contain(OpenArr,i));

OpenArr.Add(n);

}

}

}

//节点加入Closed表中 表示已经访问过了。

ClosedArr.Add(node);

//对节点进行排序

OpenArr.Sort(new MyComparer());

}

}

//理论上不可能出现这种情况

path = "Not Found @!";

return;

}

四、试验结果

1)8数问题搜索路径如下图

Generate 是用来随机产生一个初始状态(46) 表示从标准状态见过随机的46步后产生的一个随机的初始状态。这里46是一个随机数,由于该数越大离目标越远,搜索越复杂故将此随机数控制在0-100之间。3表示3的方正即8数问题,4表示4的方正即表示15数问题。

2)15数问题搜索路径如下图

从以上结果来看,由于使用了启发式的搜索过程,因此大大加快的搜索的速度以及能尽可能经过最少的路径最快到达目标状态,由于8数问题比较简单因此搜索速度较快,而15数问题复杂度加大,当随机产生的数接近100的时候搜索的时间迅速变慢,故算法还待改进,主要是因为随着深度的加深,OpenArr和CloseArr表中的数据迅速扩大,故可以考虑把OpenArr表中的状态数进行选择性的排除一些,比如每次把OpenArr中表的数据经过排序后可以删除最后的几个最差的状态,这样在一定程度上提高了速度但是也减低了找到最优解的几率不过这种减低是非常小的,因为被排除的已经是最差的情况了。

请指教,要源代码的email我。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有