我花了一夜用数据结构给女朋友写个H5走迷宫游戏 | CSDN 博文精选

戳蓝字“CSDN云计算”关注我们哦!
640?wx_fmt=jpeg
作者 | bigsai(同公众号)
转自| CSDN博客
责编 | 阿秃
先看效果图(在线电脑尝试地址http://biggsai.com/maze.html):

640?wx_fmt=gif
起因

640?wx_fmt=jpeg

又到深夜了,我按照以往在公众号写着数据结构!这占用了我大量的时间!我的超越妹妹严重缺乏陪伴而 怨气满满!

640?wx_fmt=gif

超越妹妹时常埋怨,认为数据结构这么抽象难懂的东西没啥作用,常会问道:天天写这玩意,有啥作用。而我答道:能干事情多了,比如写个小游戏啥的!

640?wx_fmt=png

当我码完字准备睡觉时:写不好别睡觉!

640?wx_fmt=gif

分析

如果用数据结构与算法造出东西来呢?

  • 什么东西简单容易呢?我百度一下,我靠,这个鸟游戏原来不好搞啊,得接触一堆不熟悉的东西,搞不来搞不来。有了(灵光一闪),写个猜数字游戏,问他加减乘除等于几。
  • 超越妹妹又不是小孩子,糊弄不过去。
  • 经过一番折腾,终于在半夜12点确定写迷宫小游戏了。大概弄清楚其中的几个步骤。

大概是:画线—>画迷宫(擦线)—>方块移动、移动约束(不出界不穿墙)—>完成游戏。

画线(棋盘)

对于html+js(canvas)画的东西,之前学过javaswing应该有点映像。在html中有个canvas 的画布,可以在上面画一些东西和声明一些监听(键盘监听)。

对于迷宫来说,那些线条是没有属性的,只有位置x,y,你操作这个画布时候,可能和我们习惯的面相对象思维不一样。所以,在你设计的线或者点的时候,记得那个点、线在什么位置,在后续划线还是擦线还是移动的时候根据这个位置进行操作。

<!DOCTYPE html><html><head><title>MyHtml.html</title></head><body><canvas id="mycanvas" width="600px" height="600px"></canvas></body><script type="text/javascript">var aa=14;var chess = document.getElementById("mycanvas");var context = chess.getContext('2d');//  var context2 = chess.getContext('2d');//      context.strokeStyle = 'yellow';var tree = [];//存放是否联通var isling=[];//判断是否相连for(var i=0;i<aa;i++){        tree[i]=[];for(var j=0;j<aa;j++){            tree[i][j]=-1;//初始值为0        }    }  for(var i=0;i<aa*aa;i++){        isling[i]=[];for(var j=0;j<aa*aa;j++){            isling[i][j]=-1;//初始值为0        }    }function drawChessBoard(){//绘画for(var i=0;i<aa+1;i++){            context.strokeStyle='gray';//可选区域            context.moveTo(15+i*30,15);//垂直方向画15根线,相距30px;            context.lineTo(15+i*30,15+30*aa);            context.stroke();            context.moveTo(15,15+i*30);//水平方向画15根线,相距30px;棋盘为14*14;context.lineTo(15+30*aa,15+i*30);            context.stroke();        }    }    drawChessBoard();//绘制棋盘//      var mymap=new Array(36);//      for(var i=0;i<36;i++)//     {mymap[i]=-1;}</script></html>

实现效果
640?wx_fmt=png
画迷宫

随机迷宫怎么生成?怎么搞?一脸懵逼。

  • 因为我们想要迷宫,那么就需要这个迷宫出口和入口有连通路径,你可能压根不知道迷宫改怎么生成,用的什么算法。小声BB:用并查集(不相交集合)。

迷宫和不相交集合有什么联系呢?(规则)

  • 之前笔者在前面数据结构与算法系列中曾经介绍过并查集(不相交集合),它的主要功能是森林的合并,不联通的通过并查集能够快速将两个森林合并,并且能够快速查询两个节点是否在同一个森林中!

而我们的随机迷宫:在每个方格都不联通的情况下,是一个棋盘方格,这也是它的初始状态。而这个节点可以跟邻居可能相连,也可能不相连。我们可以通过并查集实现。

具体思路为:(主要理解并查集)

1:定义好不想交集合的基本类和方法(search,union等)
2:数组初始化,每一个数组元素都是一个集合,值为-1
3:随机查找一个格子(一维数据要转换成二维,有点麻烦),在随机找一面墙(也就是找这个格子的上下左右),还要判断找的格子出没出界。
具体在格子中找个随机数m——>随机数m在二维中的位置[m/长,m%长]——>这个二维的上下左右随机找一个位置p[m/长+1,m%长]或[m/长-1,m%长]或[m/长,m%长+1]或[m/长,m%长-1]——>判断是否越界
4:判断两个格子(一维数组编号)是否在一个集合(并查集查找)。如果在,则重新找,如果不在,那么把墙挖去
5:把墙挖去有点繁琐,需要考虑奇偶判断它那种墙(上下还是左右,还要考虑位置),然后擦掉。(根据数组转换成真实距离)。具体为找一个节点,根据位置关系找到一维数组的号位用并查集判断是否在一个集合中。
6:最终得到一个完整的迷宫。直到第一个(1,1)和(n,n)联通停止。虽然采用随机数找墙,但是效果并不是特别差。其中要搞清一维二维数组的关系。一维是真实数据,并查集操作。二维是位置。要搞懂转化!
注意:避免混淆,搞清数组的地址和逻辑矩阵位置。数组从0开始的,逻辑上你自己判断。别搞混淆!

640?wx_fmt=png

主要逻辑为:
while(search(0)!=search(aa*aa-1))//主要思路    {var num = parseInt(Math.random() * aa*aa );//产生一个小于196的随机数var neihbour=getnei(num);if(search(num)==search(neihbour)){continue;}else//不在一个上        {           isling[num][neihbour]=1;isling[neihbour][num]=1;            drawline(num,neihbour);//划线            union(num,neihbour);        }    }

那么在前面的代码为
<!DOCTYPE html><html><head><title>MyHtml.html</title></head><body><canvas id="mycanvas" width="600px" height="600px"></canvas></body><script type="text/javascript">//自行添加上面代码//      var mymap=new Array(36);//      for(var i=0;i<36;i++)//     {mymap[i]=-1;}function getnei(a)//获得邻居号  random{var x=parseInt(a/aa);//要精确成整数var y=a%aa;var mynei=new Array();//储存邻居if(x-1>=0){mynei.push((x-1)*aa+y);}//上节点if(x+1<14){mynei.push((x+1)*aa+y);}//下节点if(y+1<14){mynei.push(x*aa+y+1);}//有节点if(y-1>=0){mynei.push(x*aa+y-1);}//下节点var ran=parseInt(Math.random() * mynei.length );return mynei[ran];    }function search(a)//找到根节点{if(tree[parseInt(a/aa)][a%aa]>0)//说明是子节点        {return search(tree[parseInt(a/aa)][a%aa]);//不能压缩路径路径压缩        }elsereturn a;    }function value(a)//找到树的大小{if(tree[parseInt(a/aa)][a%aa]>0)//说明是子节点        {return tree[parseInt(a/aa)][a%aa]=value(tree[parseInt(a/aa)][a%aa]);//不能路径压缩        }elsereturn -tree[parseInt(a/aa)][a%aa];    }function union(a,b)//合并{var a1=search(a);//a根var b1=search(b);//b根if(a1==b1){}else        {if(tree[parseInt(a1/aa)][a1%aa]<tree[parseInt(b1/aa)][b1%aa])//这个是负数(),为了简单减少计算,不在调用value函数            {                tree[parseInt(a1/aa)][a1%aa]+=tree[parseInt(b1/aa)][b1%aa];//个数相加  注意是负数相加                tree[parseInt(b1/aa)][b1%aa]=a1;       //b树成为a树的子树,b的根b1直接指向a;}else            {                tree[parseInt(b1/aa)][b1%aa]+=tree[parseInt(a1/aa)][a1%aa];                tree[parseInt(a1/aa)][a1%aa]=b1;//a所在树成为b所在树的子树            }        }    }function drawline(a,b)//划线,要判断是上下还是左右{var x1=parseInt(a/aa);var y1=a%aa;var x2=parseInt(b/aa);var y2=b%aa;        var x3=(x1+x2)/2;var y3=(y1+y2)/2;if(x1-x2==1||x1-x2==-1)//左右方向的点  需要上下划线        {//alert(x1);//  context.beginPath();            context.strokeStyle = 'white';//    context.moveTo(30+x3*30,y3*30+15);   context.lineTo(30+x3*30,y3*30+45);            context.clearRect(29+x3*30, y3*30+16,2,28);//    context.stroke();        }else        {//   context.beginPath();            context.strokeStyle = 'white';//  context.moveTo(x3*30+15,30+y3*30);    context.lineTo(45+x3*30,30+y3*30);            context.clearRect(x3*30+16, 29+y3*30,28,2);//      context.stroke();        }    }while(search(0)!=search(aa*aa-1))//主要思路    {var num = parseInt(Math.random() * aa*aa );//产生一个小于196的随机数var neihbour=getnei(num);if(search(num)==search(neihbour)){continue;}else//不在一个上        {           isling[num][neihbour]=1;isling[neihbour][num]=1;            drawline(num,neihbour);//划线            union(num,neihbour);        }    }</script></html>

实现效果:

640?wx_fmt=png

640?wx_fmt=png

方块移动

这部分我采用的方法不是动态真的移动,而是一格一格的跳跃。也就是当走到下一个格子将当前格子的方块擦掉,在移动的那个格子中再画一个方块。选择方块是因为方块更方便擦除,可以根据像素大小精准擦除。

另外,再移动中要注意不能穿墙、越界。那么怎么判断呢?很好办,我们再前面会判断两个格子是否联通,如果不连通我们将把这个墙拆开。再拆的时候把这个墙的时候记录这两点拆墙可走即可(数组)

另外,事件的监听上下左右查一查就可以得到,添加按钮对一些事件监听,这些不是最主要的。

为了丰富游戏可玩性,将方法封装,可以设置关卡(只需改变迷宫大小)。这样就可以实现通关了。另外,如果写成动态存库那就更好了。

640?wx_fmt=png

结语

在线尝试地址,代码直接查看网页源代码即可!
640?wx_fmt=png

扫描下方二维码,看作者大大原文~

640?wx_fmt=png
640?wx_fmt=jpeg

福利
扫描添加小编微信,备注“姓名+公司职位”,入驻【CSDN博客】,加入【云计算学习交流群】,和志同道合的朋友们共同打卡学习!

640?wx_fmt=jpeg

推荐阅读:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/520496.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

异常:This application has no explicit mapping for /error, so you are seeing this as a fallback.

初学springboot配置好文件,测试接口出现这个是springboot的启动器要放在dao service…包的上层目录

安全看得见,阿里云性能监控 ARMS 全真3D拓扑实现一“屏”了然

微服务架构下&#xff0c;各类服务之间存在着错综复杂的依赖关系。一旦业务出现问题&#xff0c;追查问题源头就好比大海捞针&#xff0c;没有头绪。但业务不等人&#xff0c;此时&#xff0c;在最短的时间内定位问题根源是开发和运维人员对微服务监控产品的核心诉求。 传统的…

z变换判断稳定性和因果性_图像处理的仿射变换与透视变换

原文首发于微信公众号&#xff1a;【3D视觉工坊】。引言这一周主要在研究图像的放射变换与透视变换&#xff0c;目前出现的主要问题是需要正确识别如下图中的编码标志点圆心。1.当倾斜角较小时&#xff1a;倾斜角较小2.倾斜角较大时&#xff1a;倾斜角较大由上面两幅图可以看出…

阿里云ACM:云原生配置管理利器,让云上的Spring Cloud应用配置管理舞动起来

在传统架构中&#xff0c;如果配置信息有变更&#xff0c;通常是登陆服务器手动修改配置来使配置生效。在微服务架构中&#xff0c;应用数和节点数由于微服务化数量激增&#xff0c;导致发布次数增加&#xff0c;配置变更难度加大&#xff0c;通常是将应用配置抽象出来放置在外…

技术和商业的碰撞,谈阿里云与天猫双11这十年

2009年&#xff0c;发生了两件看似不起眼的事。 初春刚过&#xff0c;阿里云在北京一栋没有暖气的写字楼写下了飞天第一行代码。 同年11月11日&#xff0c;淘宝商城启动了一个叫做双11的促销活动。 谁也没想到&#xff0c;多年以后他们会是现在这模样。 前传 2007年淘宝的…

windows环境 wildfly-10.1.0.Final 安装、配置、部署

文章目录一、下载、解压、配置1. 下载Wildfly2. 解压&#xff0c;成功得到文件夹。3. 配置环境变量4. 添加管控台用户5. 启动并登录管控台测试6. 管控台页面简介7. 修改默认端口由于Wildfly是依赖与Java的Jdk的&#xff0c;所以在这之前要先安装JDK。一、下载、解压、配置 1. …

揭秘 | 双11逆天记录背后的数据库技术革新

每一个数字背后 都需要强大的技术支撑 Higher, Faster, Smarter 是我们不变的追求 技术无边界 创新无止境 ▽ 关于数据库的双11 也许你还想知道 ▽ 2135亿&#xff01;2018 双11阿里数据库技术战报“双11”十年记 阿里数据库演绎变迁三部曲 原文链接 本文为云栖社区原创内容…

重磅!腾讯助推十亿级节点图计算进入分钟级时代;沈向洋离开微软;阿里:拥有超6000项已授权专利,涉及云、AI等领域……...

戳蓝字“CSDN云计算”关注我们哦&#xff01; 嗨&#xff0c;大家好&#xff0c;重磅君带来的【云重磅】特别栏目&#xff0c;如期而至&#xff0c;每周五第一时间为大家带来重磅新闻。把握技术风向标&#xff0c;了解行业应用与实践&#xff0c;就交给我重磅君吧&#xff01;重…

2135亿背后的双11项目协作怎么玩?

2018天猫双11已经收官&#xff0c;2135亿元的成交额再度刷新纪录&#xff0c;这是一场阿里巴巴经济体的深度协作。 在这个大协作项目中&#xff0c;项目协作难题难以想象&#xff1a;如何保证众多部门、百个核心产品、千个垂直项目、几千人的的高效协作&#xff1f;如何在双11…

下一代大数据处理引擎,阿里云实时计算独享模式重磅发布

11月14日&#xff0c;阿里云重磅发布了实时计算独享模式&#xff0c;即用户独享一部分物理资源&#xff0c;这部分资源在网络/磁盘/CPU/内存等资源上跟其他用户完全独立&#xff0c;是实时计算在原有共享模式基础上的重大升级。 独享模式优点更加突出 1、UDX开放&#xff1a;实…

刷爆了!李彦宏:这类程序员我给100万!你怎么看?

从2017年开始&#xff0c;人工智能便波澜不断&#xff0c;无论是从BAT高调布局AI&#xff0c;还是从年薪50万招聘AI应届生&#xff0c;炽手可热形容AI工程师一点都不过分。百度推出“少帅计划”,针对30岁以下的深度学习科学家&#xff0c;开出100万以上年薪&#xff01;阿里巴巴…

Element-UI中Cascader 级联选择器使用

Element-UI的级联组件官方文档 <el-cascaderv-model"value":options"options":props"{ expandTrigger: hover }"change"handleChange"></el-cascader>说明&#xff1a; options:绑定数据源 props:数据配置项 v-model“val…

DRDS SQL 审计与分析——全面洞察 SQL 之利器

背景 数据库存储着系统的核心数据&#xff0c;其安全方面的问题在传统环境中已经成为泄漏和被篡改的重要根源。而在云端&#xff0c;数据库所面临的威胁被进一步的放大。因此&#xff0c;对云数据库的操作行为尤其是全量 SQL 执行记录的审计日志&#xff0c;就显得尤为重要&am…

机器学习:从入门到晋级

目前&#xff0c;人工智能&#xff08;AI&#xff09;非常热门&#xff0c;许多人都想一窥究竟。如果你对人工智能有所了解&#xff0c;但对机器学习&#xff08;Machine Learning&#xff09;的理解有很多的困惑&#xff0c;那么看完本文后你将会对此有进一步深入理解。在这里…

漫画 | 面试的我 VS 真实的我

戳蓝字“CSDN云计算”关注我们哦&#xff01;作者&#xff5c;纯洁的微笑、孤独烟责编&#xff5c;阿秃“面试造火箭&#xff0c;入职拧螺丝&#xff01;”已经是各大互联网公司招聘的常态&#xff0c;为了应对如今越演越烈的面试形势&#xff0c;程序员一个个都变成了表演大师…

Visual Studio Code Vue代码片段 总览

文章目录一、模板篇1. 初始化.vue 文件模板2. template 作用域插槽模板二、响应提示篇1. error 响应 &#xff01;200提示2. success 响应 200提示三、axios请求篇1. axios get 请求2. axios post 请求3. axios put 请求4. axios delete请求四、事件篇…

一个案例彻底弄懂如何正确使用 mysql inndb 联合索引

有一个业务是查询最新审核的5条数据 SELECT id, title FROM th_content WHERE audit_time < 1541984478AND status ONLINE ORDER BY audit_time DESC, id DESC LIMIT 5; 查看当时的监控情况 cpu 使用率是超过了100%&#xff0c;show processlist看到很多类似的查询都是处…

为你的AliOS Things应用增加自定义cli命令

在日常嵌入式开发中&#xff0c;我们经常会用串口命令来使设备进入某种特定的状态&#xff0c;或执行某个特定的操作。如系统自检&#xff0c;模拟运行&#xff0c;或者进入手动模式进行设备点动。linux下有强大的shell工具&#xff0c;可以让用户和片上系统进行交互&#xff0…

4________请拖拽到此区域5________请拖拽到此区域_5分钟学会:蒙版与通道(剪贴蒙版)...

剪贴蒙版可以用一个图层中包含像素的区域来限制它上层图像的显示范围。它的最大优点是可以通过一个图层来控制多个图层的可见内容&#xff0c;而图层蒙版和矢量蒙版都只能控制一个图层。1、实例&#xff1a;创建剪贴蒙版颜回&#xff1a;5分钟学会&#xff1a;蒙版与通道 实例&…

一场高质量的技术盛会怎样炼成?「2019中国大数据技术大会」即将来临,邀您共赴!...

2019年12月&#xff0c;一场轰动国内产业界、学术界、科研界及投资领域的顶级科技盛会即将拉开帷幕&#xff0c;它涵盖大数据、人工智能、云计算、AIoT、金融科技、智能制造等十几个前沿领域的热门话题。在过去十二年里&#xff0c;这场盛会从最初仅 60 余人参加的技术沙龙到如…