使用 C# 开发智能手机软件:推箱子(四)
这是“使用
C# 开发智能手机软件:推箱子”系列文章的第四篇。
在这篇文章中,介绍 Common/FindPath.cs 源程序文件。
using System.Drawing;
using System.Collections.Generic;
namespace Skyiv.Ben.PushBox.Common
{
/// <summary>
/// 寻找最短路线
/// </summary>
static class FindPath
{
static Size[] offsets = { new Size(0, 1), new Size(1, 0), new Size(0, -1), new Size(-1, 0) };
static Direction[] directions = { Direction.South, Direction.East, Direction.North, Direction.West };
/// <summary>
/// 寻找最短路线
/// </summary>
/// <param name="map">地图</param>
/// <param name="from">出发点</param>
/// <param name="to">目的地</param>
/// <returns>最短路线</returns>
public static Queue<Direction> Seek(ushort[,] map, Point from, Point to)
{
Queue<Direction> moveQueue = new Queue<Direction>(); // 路线
int value; // 离目的地距离
if (Seek(map, to, out value)) // 找到了一条路线
{
Point here = from; // 出发点(即工人的位置)
Point nbr = new Point(); // 四周的邻居
for (value--; value > 0; value--) // 逐步走向目的地
{
for (int i = 0; i < offsets.Length; i++)
{
nbr = Fcl.Add(here, offsets[i]); // 開始寻找四周的邻居
if (Block.Value(map[nbr.Y, nbr.X]) == value) // 就往这个方向走
{
moveQueue.Enqueue(directions[i]); // 路线向目的地延伸一步
break;
}
}
here = nbr; // 继续前进
}
}
Block.CleanAllMark(map); // 清除全部标志,恢复现场
return moveQueue; // 所寻找的路线,假设无法到达目的地则为该路线的长度为零
}
/// <summary>
/// 寻找最短路线,使用广度优先搜索
/// </summary>
/// <param name="map">地图</param>
/// <param name="to">目的地</param>
/// <param name="value">输出:路线的长度(加1)</param>
/// <returns>是否成功</returns>
static bool Seek(ushort[,] map, Point to, out int value)
{
Queue<Point> q = new Queue<Point>();
Block.Mark(ref map[to.Y, to.X], 1); // 从目的地開始往回寻找出发点,目的地标记为1
Point nbr = Point.Empty; // 四周的邻居
for (; ; )
{
value = Block.Value(map[to.Y, to.X]) + 1; // 离开目的地的距离(加1),用作标记
for (int i = 0; i < offsets.Length; i++)
{
nbr = Fcl.Add(to, offsets[i]); // 開始寻找四周的邻居
if (Block.IsMan(map[nbr.Y, nbr.X])) break; // 到达出发点(即工人的位置)
if (Block.IsBlank(map[nbr.Y, nbr.X])) // 能够走的路
{
Block.Mark(ref map[nbr.Y, nbr.X], value); // 标记,防止以后再走这条路
q.Enqueue(nbr); // 增加队列,等待以后继续寻找
}
}
if (Block.IsMan(map[nbr.Y, nbr.X])) break; // 到达出发点
if (q.Count == 0) return false; // 无法到达出发点
to = q.Dequeue(); // 出队,继续寻找,这是广度优先搜索,由于前面已经把四周可以走的路所有增加队列中了.
}
return true; // 找到一条路线
}
}
}
静态类 FindPath 是用来寻找工人移动到鼠标点击的目的地的最短路线的。她採用一种广度优先搜索算法,使用循环,没有使用递归,并且地图上已经搜索过的路线决不再走第二遍。
该算法分两个阶段进行:首先是寻找并标记最短路线,由该类的第二个 Seek 方法实现。这个私有的方法返回一个布尔值表明是否成功。
然后,假设在第一阶段中找到了一条路线,则依据第一阶段所做的标记生成最短路线并将该路线返回给调用者。我们来看几个实例:
在该算法中,是从要到达的目的地開始往回寻找出发点。首先,将目的地标记为1,然后查看周围的四个邻居(按南、东、北、西的顺序)是否是“空白”(即“地”和“槽”,使用 Block.IsBlank 方法来推断),如是,则表明这是能够走的路,将其作上标记(使用 Block.Mark 方法,标记的数值等于离开目的地的距离加一),然后增加队列。这有两个作用,首先,标记过的单元格将不再被觉得是能够走的路。防止反复搜索。
其次,在第二阶段中要依据标记的值来生成最短路线。
假设发现周围的邻居中有一个是工人(用
Block.IsMan 方法来推断),说明到达出发点,则马上结束搜索,退出循环。返回成功。
否则,就检查队列是否为空,假设为空,则说明无法到达出发点,返回失败。假设不为空,则出队,从这一点继续開始搜索。
如此一直循环。
这个算法是广度优先的,如上面的两个图所看到的。该算法是按标记的值从小到大进行遍历的,而该标记的值表示的是离开目的地的距离加一。
第二个阶段。假设在第一阶段返回失败,则返回一条空的路线(长度为零)给调用者。否则,从出发点(即工人的位置)開始,查看周围的四个邻居(按南、东、北、西的顺序)。假设其标记的值(使用 Block.Value 方法来获得)为到目的地的距离加一(至少能够找到一个,可能有多个。能够任取一个,程序中使用第一个),就往这个方向走。
如此一直循环。直到到达目的地。然后返回这条路线给调用者。
从这里能够看出。为什么地图(即 ushort[,] map)要使用 ushort 而不使用 byte。由于在该算法须要在地图中作标记,并且标记的值还必须是到目的地的距离加一(假设仅仅须推断目的地是否可达,而不要求给出到达目的地的详细路线,则在算法中标记的值可所有都为1。这样用 byte 就足够了)。地图中总共同拥有八种类型的单元格,须要用三个二进位表示。而
byte 仅仅有八个二进位。那么。仅仅剩下五个二进位,25=32。也就是说,目的地在工人32步以外该算法就无能为力了。而 ushort 有十六个二进位,减去三个,还有十三个二进位,213=8192。这应该足够了。让我们看看下图吧:
这是一个 9x9 的地图。共同拥有81个单元格,当中49个是空地。如果目的在地图的右上角(标记为1的地方),则工人须要48步才干到达目的地。依据计算。如果是 NxN (这里N是奇数)的地图,工人在最坏的情况下须要 (N2 -
1)/2 + N -1 步(走最短路线)才干到达目的地。
这就是说,在 127x127 的地图上,工人最多仅仅须要 8190 步就能够到达目的地。这刚好在我们算法的范围之内。假设地图再大。我们的算法就可能(仅仅是可能,由于在大地图上普通情况下并不会出现超过 8192 步的最短路线)无能为力了。
使用 C# 开发智能手机软件:推箱子(四)的更多相关文章
- 使用 C# 开发智能手机软件:推箱子(十四)
这是"使用 C# 开发智能手机软件:推箱子"系列文章的第十四篇.在这篇文章中,介绍 Window/ErrorMsgDlg.cs 源程序文件.这个源程序文件包括 ErrorMsgDl ...
- 使用 C# 开发智能手机软件:推箱子(十二)
这是"使用 C# 开发智能手机软件:推箱子"系列文章的第十二篇.在这篇文章中,介绍 Window/AboutDlg.cs 源程序文件. 这个源程序文件包括 AboutDlg 类,该 ...
- 使用 C# 开发智能手机软件:推箱子(二)
在上篇文章"使用 C# 开发智能手机软件:推箱子(一)"中.我对推箱子程序作了整体介绍.这次,我先介绍 Common/Fcl.cs 源程序文件. 1 using System; ...
- 使用 C# 开发智能手机软件:推箱子(十八)
这是"使用 C# 开发智能手机软件:推箱子" 系列文章的第十八篇.在这篇文章中.介绍 Window/SelectLevelDlg.cs 源程序文件. 这个源程序文件包括 Selec ...
- 使用 C# 开发智能手机软件:推箱子(三)
这是"使用 C# 开发智能手机软件:推箱子"系列文章的第三篇.在这篇文章中,介绍 Common/Block.cs 源程序文件. 1 namespace Skyiv.Ben.Pu ...
- [转]Flash ActionScript2.0面向对象游戏开发-推箱子
本文转自:http://www.alixixi.com/Dev/W3C/Flash/2007/2007070868666.html 概述: Flash ActionScript2.0是一种面向对向的编 ...
- 每个人都可以用C语言写的推箱子小游戏!今天你就可以写出属于自己项目~
C语言,作为大多数人的第一门编程语言,重要性不言而喻,很多编程习惯,逻辑方式在此时就已经形成了.这个是我在大一学习 C语言 后写的推箱子小游戏,自己的逻辑能力得到了提升,在这里同大家分享这个推箱子小游 ...
- 用C写一个简单的推箱子游戏(一)
我现在在读大二,我们有一门课程叫<操作系统>,课程考查要求我们可以写一段程序或者写Windows.iOS.Mac的发展历程.后面我结合网上的资料参考,就想用自己之前简单学过的C写一关的推箱 ...
- JavaScript写一个小乌龟推箱子游戏
推箱子游戏是老游戏了, 网上有各种各样的版本, 说下推箱子游戏的简单实现,以及我找到的一些参考视频和实例: 推箱子游戏的在线DEMO : 打开 如下是效果图: 这个拖箱子游戏做了移动端的适配, 我使用 ...
随机推荐
- xhtml css 漏 整理
1)文档类型 代码最上部有如下这句话: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" &quo ...
- spring配置datasource三种方式 数据库连接池
尊重原创(原文链接):http://blog.csdn.net/kunkun378263/article/details/8506355 1.使用org.springframework.jdbc.da ...
- C/C++复杂类型声明
曾经碰到过让你迷惑不解.类似于int * (* (*fp1) (int) ) [10];这样的变量声明吗?本文将由易到难,一步一步教会你如何理解这种复杂的C/C++声明. 我们将从每天都能碰到的较 ...
- 4A. Just a Hook
4A. Just a Hook Time Limit: 2000ms Case Time Limit: 2000ms Memory Limit: 32768KB 64-bit integer IO ...
- [UOJ#223][BZOJ4654][Noi2016]国王饮水记
[UOJ#223][BZOJ4654][Noi2016]国王饮水记 试题描述 跳蚤国有 n 个城市,伟大的跳蚤国王居住在跳蚤国首都中,即 1 号城市中.跳蚤国最大的问题就是饮水问题,由于首都中居住的跳 ...
- BZOJ 3529 [Sdoi2014]数表 ——莫比乌斯反演 树状数组
$ans=\sum_{i=1}^n\sum_{j=1}^n\sigma(gcd(i,j))$ 枚举gcd为d的所有数得到 $ans=\sum_{d<=n}\sigma(d)*g(d)$ $g(d ...
- BZOJ 3240 [Noi2013]矩阵游戏 ——费马小定理 快速幂
发现是一个快速幂,然而过不去. 怎么办呢? 1.十进制快速幂,可以用来练习卡时. 2.费马小定理,如果需要乘方的地方,可以先%(p-1)再计算,其他地方需要%p,所以需要保存两个数. 然后就是分类讨论 ...
- Mychael原创题 洛谷T23923 Mychaelの水题 【题解】
原题链接 题目大意: 有来自三个地区的人各a,b,c位,他们排成了一排.请问有多少种不同类型的排法,使得相邻的人都来自不同的地区 \(a,b,c<=200\) 答案取模 题解 弱弱的标程解法 设 ...
- out.print和out.write
这是一个JSP页面: <%@ page language="java" import="java.util.*" %> <%@ page p ...
- 巧克力王国 BZOJ 2850
巧克力王国 [问题描述] 巧克力王国里的巧克力都是由牛奶和可可做成的.但是并不是每一块巧克力都受王国人民的欢迎,因为大家都不喜欢过于甜的巧克力.对于每一块巧克力,我们设x和y为其牛奶和可可的含量.由于 ...