《C++游戏开发》十六 游戏中的寻路算法(二):迷宫&A*算法基础
本系列文章由七十一雾央编写,转载请注明出处。
http://blog.csdn.net/u011371356/article/details/10289253
作者:七十一雾央 新浪微博:http://weibo.com/1689160943/profile?rightmod=1&wvr=5&mod=personinfo
因为前段时间在学习Cocos2d-X引擎,然后自己最近就练手写了个小游戏练习,花了自己不少时间,所以这个系列没怎么更新,不过以后雾央会继续更新的,分享自己学到的新东西。
上一节本来雾央说要先讲迷宫,但是至少在现在,雾央觉得迷宫用处不是太大,所以就不打算详细写了,这里只是简略的介绍一下吧。
在以前的游戏中,由于硬件性能的原因导致不能负担起丰富的画面,同时也为了减轻美术人员的工作量,如何利用少数的资源创造出不同的游戏是一个很值得探讨的问题,前辈游戏程序员们给出的答案就是迷宫。使用迷宫,一方面可以利用几面墙就可以采用算法随机生成无穷无尽的地图,使玩家得到不同的体验,另一方面,也可以大大延长游戏的时长,毕竟,大家不希望自己画几十块钱买来的游戏一会就通关了吧。
对于迷宫,涉及到的一个问题就是随机生成算法。
大家可以首先将迷宫看成一个个方格,比如下图
我们可以规定一个入口和一个出口,比如左上角的格子和右下角的格子
现在我们需要的就是有一条路连接入口和出口,一般情况下都是要求路线是唯一的。
如果我们把每个格子都看作一个点,那么它实际上就是一个图。问题也就是等价于找到这张图的生成树。那么涉及到生成树的算法我们都可以拿来用了,比如BFS,DFS,Prim,Kruskal等等
对这些感兴趣的朋友,可以看这篇文章
讲解的非常好
它里面还提到一种巧妙的方法,递归分割,感觉很巧,大家可以看看。
另外,在学数据结构的时候,我们学习过并查集,也叫不相交集,利用并查集也可以得到一种不错的算法。一开始每个格子都是一个集合,然后随机的选择一面墙将它拆掉,将墙两边的格子连接起来,合并它们俩所在的集合,判断出口和入口格子是不是在同一个集合中,如果不在则重复拆墙合并过程。
迷宫的另外一个问题就是寻路算法
高端的算法当然都是适用的,但是一般迷宫只是为了寻找一条通路而已,常用的DFS,BFS的就可以了,容易理解,也容易实现。
今天主要是想说一下雾央最近学习的A*算法,雾央也是个算法渣,但是最近迫切的感觉到了寻路算法的重要性,所以还是要认真学习下,网上有很多人写的A*算法,都很不错,下面写的都是雾央的理解,希望可以尽可能通俗。
最近雾央写了一个小游戏,其中的普通地图如下:
大家看到是采用摇杆来控制运动的,雾央是把这个游戏移植到Android上了,所以大家肯定是会觉得很蛋疼,那么小的屏幕上怎么好控制!其实雾央也不想这样的,直接点哪人物走哪不是很好吗,可是那样就涉及到一个问题了:寻路问题!地图上是有障碍物的,点击了一个地方后,人物肯定要以最短路径走过去,这是雾央学习中的第一个Android游戏,就没想给自己找麻烦,所以采用了蛋疼的摇杆操作。
用安卓机的朋友们,如果感兴趣,可以感兴趣试玩一下,呵呵,详细请点击这篇文章点我啊。
扯上面这些其实是想说,寻路算法还是挺重要的,今天只说最基础的A*算法,没有任何变异的品种。
由于A*算法的重要性和基础性,网上对A*算法的研究比比皆是,有很多大神都会他们进行过各种讨论和延伸。但是很多对于我们初学者来说并不合适,下面这个链接是对A*算法最基础最明了的说明,英文好的朋友可以直接看这篇文章
英文不好的朋友也可以看这篇中文版,翻译的还行,大家应该也可以看懂。
大家看上面这两篇应该就可以了,下面的是雾央学习A*算法对它的理解,再叙述一遍加深自己对它的理解,希望可以更通俗一点,有兴趣的朋友可以参考下。
问题与目标
首先明确一下希望解决的问题,我们看一下下面这张图。
这张图就代表我们的地图,为了方便,我们用方格代替,当然它也可以是多边形神马的。
绿色方格代表起点,红色方格代表终点,蓝色方格代表障碍物。
我们的目标就是找到一条最短路径从绿色方格到达红色方格。
路径评估方案
我们在寻找路径的时候会遇到很多种选择,那么那一条是最好的呢?这需要我们做出判断,也就需要一个标准。我们采用的评判公式是
F=G+H
下面雾央来具体解释一下这个公式的含义
F代表这条路径的长度,我们就希望找到具有最小F值的那条路
我们在寻路时从起点A出发是要经过一系列方格到达终点的,比如我们中途到达了某一结点M点,那么从起点A到结点M的路径已知,就是这里的G值啦
但是从M点到终点B的过程又是这样的一个寻路问题,我们不知道它的长度,所以需要采用估算算法,它的估计值就是H了。
剩下的下次再写,等会要睡觉了。
另外要开学了,最近雾央也很忙,因为雾央也在尝试着做自己的游戏,也在不断的学习,所以更新的没有七月份那么及时了。这个博客系列主要就是分享雾央学习的东西,大家初学游戏也需要自己思考怎么实现一些东西,其实到现在这些东西,已经足够大家做出自己的小游戏了,建议大家可以尝试做一下了,看似很复杂的游戏逻辑可以一点点写,慢慢组装起来,等以后再考虑架构可维护性神马的问题。先实现功能,在考虑改进优化的问题。在微博上有好几位多朋友问雾央学习的过程,其实雾央到现在也是一个初学者,到现在雾央学习过的东西都写在这篇文章中了,有兴趣的朋友可以看看,希望对你有一些参考意义,雾央学过的东东,其实更重要的是自己动手实践,写几个小游戏慢慢就会成长。
《C++游戏开发》十六 游戏中的寻路算法(二):迷宫&A*算法基础的更多相关文章
- cocos2d-x游戏开发(十六)帧动画
欢迎转载:http://blog.csdn.net/dawn_moon/article/details/11775745 本来想写一下帧动画的,搜了一下网上好像有一大把,就懒得写了,直接贴代码. // ...
- cocos2d-x游戏开发(十五)游戏加载动画loading界面
个人原创,欢迎转载:http://blog.csdn.net/dawn_moon/article/details/11478885 这个资源加载的loading界面demo是在玩客网做逆转三国的时候随 ...
- cocos2d-x游戏开发(十五)游戏载入动画loading界面
这个资源载入的loading界面demo是在玩客网做逆转三国的时候随手写的,尽管我在那仅仅待了2个礼拜.可是也算參与了一个商业游戏项目了,学到不少东西.当时使用的cocos2d-x还是1.0版的,我用 ...
- [Unity游戏开发]向量在游戏开发中的应用(二)
本文已同步发表在CSDN:http://blog.csdn.net/wenxin2011/article/details/50972976 在上一篇博客中讲了利用向量方向的性质来解决问题.这篇博客将继 ...
- Unity 2D游戏开发教程之游戏中精灵的跳跃状态
Unity 2D游戏开发教程之游戏中精灵的跳跃状态 精灵的跳跃状态 为了让游戏中的精灵有更大的活动范围,上一节为游戏场景添加了多个地面,于是精灵可以从高的地面移动到低的地面处,如图2-14所示.但是却 ...
- 【转载】【游戏开发】在Lua中实现面向对象特性——模拟类、继承、多态
[游戏开发]在Lua中实现面向对象特性——模拟类.继承.多态 阅读目录 一.简介 二.前提知识 三.Lua中实现类.继承.多态 四.总结 回到顶部 一.简介 Lua是一门非常强大.非常灵活的脚本语 ...
- java游戏开发杂谈 - 实现游戏主菜单
经常玩游戏的同学,大家都知道,游戏都会有个主菜单,里面有多个菜单选项:开始游戏.游戏设置.关于游戏.退出游戏等等,这个菜单是怎么实现的呢. 有一定桌面软件开发基础的同学可能会想到,用JButton组件 ...
- J2EE进阶(十六)Hibernate 中getHibernateTemplate()方法使用
J2EE进阶(十六)Hibernate 中getHibernateTemplate()方法使用 spring 中获得由spring所配置的hibernate的操作对象,然后利用此对象进行,保存,修 ...
- Unity 2D游戏开发教程之游戏精灵的开火状态
Unity 2D游戏开发教程之游戏精灵的开火状态 精灵的开火状态 “开火”就是发射子弹的意思,在战争类型的电影或者电视剧中,主角们就爱这么说!本节打算为精灵添加发射子弹的能力.因为本游戏在后面会引入敌 ...
- [Unity3D]Unity3D游戏开发Lua随着游戏的债券(在)
---------------------------------------------------------------------------------------------------- ...
随机推荐
- ZeptoLab Code Rush 2015 B. Om Nom and Dark Park DFS
B. Om Nom and Dark Park Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/5 ...
- PHP通过AJAX及Access-Control-Allow-Origin实现跨域访问
这里的跨域实质上是由浏览器同源策略限制的一类请求场景,浏览器同源策略SOP(Same origin policy)是一种约定,由Netscape公司1995年引入浏览器,它是浏览器最核心也最基本的安全 ...
- 重温PHP之插入排序
插入排序基本思路:将数组分为两个区(已排序区和未排序区),假定数组的第一个元素处于已排序区, 第一个元素之后的所有元素都处于未排序部分.排序时用到双层循环,外层循环用于从未排序部分中取出待排序元素,并 ...
- wamp经典安装
1,根据综述 本机 注意,现在apache2.2不能和5.5php, 2.4apache和5.5php就可以 window10 64位 vc14 apache 2.4.23 注意,虚拟 ...
- Tasker to stop Poweramp control for the headset while there is an incoming SMS - frozen
If you usually like to use Poweramp or any other media player to enjoy the music with headset plugge ...
- GCC 对C语言的扩展
http://www.cnblogs.com/emituofo/archive/2012/07/20/2600995.html http://blog.csdn.net/andyhuabing/art ...
- [转载] MATLAB快捷键
原文地址,点此查看 一.常用对象操作 除了一般windows窗口的常用功能键外. 1.!dir 可以查看当前工作目录的文件. !dir& 可以在dos状态下查看. 2.who 可以查看当前 ...
- C#编程(八)--------- Main函数
Main()方法. C#程序是以Main()开始执行的,这个方法必须是类或结构的静态方法,并且其返回类型必须是int或者void. 虽然显示指定public修饰符很常见,但是我们也可以把该方法标记为p ...
- 【C#高级编程】笔记之核心C#
Main()方法 每一个C#可执行文件(如控制台程序.Windows程序和Windows服务)都必须有一个入口点——Main()方法(注意M大写). 这个方法必须是类或静态方法,并且返回类型必须是in ...
- Android 性能优化之使用MAT分析内存泄露
转载请注明本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/42396507),请尊重他人的辛勤劳动成果,谢谢! 我们平常 ...