JavaScript面向对象编程小游戏---贪吃蛇代码实例


Posted in Javascript onMay 15, 2019

1 面向对象编程思想在程序项目中有着非常明显的优势:
1- 1 代码可读性高.由于继承的存在,即使改变需求,那么维护也只是在局部模块
1-2 维护非常方便并且成本较低。

​2 这个demo是采用了面向对象的编程思想. 用JavaScript 语言编写的游戏小程序--贪吃蛇.
​ 代码注释详细,逻辑清晰 . 非常适合新手前端开发者, 锻炼JavaScript语言的面向对象的编程思想. 
该小Demo已上传GitHub,欢迎下载!  觉得好的话,随手给个star,  您的star是我最大的动力!

https://github.com/XingJYGo/snakePlay#javascript-经典面向对象demo-贪吃蛇

代码展示:
代码结构:
--index.html 地图页面,展示蛇和食物,进行游戏
--food.js   构造食物对象
--game.js   构造游戏对象
--snake.js   构造蛇对象
--tool.js   常用数据工具封装

--index.html 地图页面

<!DOCTYPE html>
<html lang="zh-CN">
<head>
 <meta charset="UTF-8">
 <title>Title</title>
 <style>
  #map{
   width: 500px;
   height: 500px;
   background-color: lightblue;
   position: relative;
  }
  
 </style>
</head>
<body>
<div id="map">
 
</div>
<button id="btn">模拟蛇吃到食物</button>
<script src="tool.js"></script>
<script src="food.js"></script>
<script src="snake.js"></script>
<script src="game.js"></script>
<script>
 
////==========前后方向设定后 ==================
var game = new Game();
game.start();
 
 
 
</script>
 
</body>
</html>

--food.js   构造食物对象

// 封装一个食物对象
//沙箱模式
(function(){
 var container; //用于存储之前的食物
 function Food(option) {
  //防止用户不传参数会报错
  option = option || {};
  this.width = option.width || 20;
  this.height = option.height || 20;
  this.bgc = option.bgc || 'orange';
  this.x = option.x || 0;
  this.y = option.y || 0;
  this.borderRadius = option.borderRadius |10;
 }
 
 Food.prototype.render = function () {
  //每一次渲染新的之前就把原来的移除掉
  if(container){
   map.removeChild(container);
  }
  // 创建食物对象
  var food = document.createElement('div');
  //存到全局变量里
  container = food;
  food.style.width = this.width + 'px';
  food.style.height = this.height + 'px';
  food.style.backgroundColor = this.bgc;
  food.style.position = 'absolute';
   //获得随机位置
   //由于要让食物的位置在每一个格子里面,所有获取随机数的算法要重新计算
  this.x = Tool.getRandom(0, (map.offsetWidth/ this.width-1)) * this.width;
  this.y = Tool.getRandom(0, (map.offsetHeight/ this.height-1)) * this.height;
  food.style.left = this.x + 'px';
  food.style.top = this.y + 'px';
  food.style.borderRadius = this.borderRadius + 'px';
  //渲染上食物
  map.appendChild(food);
 }
 
 //因为要在全局使用Food,需要把Food拿到全局中
 window.Food = Food;
})();

--game.js   构造游戏对象

(function () {
//  1 由于游戏对象要控制蛇和食物,
//   所以游戏对象应该拥有蛇的实例和食物的实例
  //存储定时器的id
  var timeid;
  function Game() {
    this.snake = new Snake();
    this.food = new Food();
  }
 
  //2 开始游戏
  Game.prototype.start = function () {
    this.snake.render();
    this.food.render();
 
    // 2-1 游戏一开始,蛇和食物就渲染出来
    timeid = setInterval(function () {
      //2-2 -1 蛇的数据改变
      this.snake.move();
      // 2-3 判断蛇是否到达边界
      var snakeHead = this.snake.body[0];
      //2-3-1 求蛇头可以移动的水平/垂直坐标的最大位置
      var maxX = map.offsetWidth/this.snake.width -1;
      var maxY = map.offsetHeight/this.snake.height -1;
      if (snakeHead.x <0 ||snakeHead.x > maxX ||snakeHead.y <0 ||snakeHead.y > maxY){
        clearInterval(timeid);
        alert("gave over");
        //注:当X超出范围,代码应立即终止,
        // 防止2-2-2 渲染出下一个盒子.展示出来
        return;
      }
 
 
      //2-4 蛇吃食物
      //依据: 蛇头的坐标 和 食物的坐标重合
      var snakeX = snakeHead.x * this.snake.width;
      var snakeY = snakeHead.y * this.snake.height;
      var foodX = this.food.x;
      var foodY = this.food.y;
 
      //如果符合条件, 证明吃到了食物
       if (snakeX === foodX && snakeY === foodY){
         // 2-4-1 食物消失, 渲染新食物
         this.food.render();
         // 2-4-2 蛇,变长
         //  其实就是往snake.body.push个新对象
         //  bug: 为了解决新添加蛇节闪下的问题, 把蛇的最后一节对象,作为新的对象.
         var last = this.snake.body[this.snake.body.length -1];
         this.snake.body.push({
           x:last.x,
           y:last.y,
           col:last.col
         })
         // this.snake.body.push(last);
         // 注:last本身已经在数组中了,
       }
      //2-2 -2渲染到页面上,真正看到的蛇动起来
      this.snake.render();
 
    }.bind(this), 150)
 
    // 3 给页面注册键盘按下的事件
    // 3-1 监听用户是否按下了上,下,左,右的按键
 
    document.onkeydown = function(e){
      // console.log(this);
      e = e || window.event;
      console.log(e.keyCode);
      // 左37 上38 右39  下40
      switch(e.keyCode){
 
        case 37:
          //3-11 需要找到蛇,修改蛇的direction属性
          //防止原地掉头
          if(this.snake.direction === 'right'){
            return;
          }
          this.snake.direction = 'left';
          break;
        case 38:
          if(this.snake.direction === 'bottom'){
            return;
          }
          this.snake.direction = 'top';
          break;
        case 39:
          if(this.snake.direction === 'left'){
            return;
          }
          this.snake.direction = 'right';
          break;
        case 40:
          if(this.snake.direction === 'top') return; //如果if中只有一行代码就可以不写花括号,然后这一行代码要紧跟在if后面记得加分号
          this.snake.direction = 'bottom';
          break;
 
      }
    }.bind(this);
 
 
 
  };
 
  //2-2 蛇变量赋予全局
  window.Game = Game;
 
})();

--snake.js   构造蛇对象

(function () {
  var arr = []; //用于存储蛇的每一节数据
 
  // 1 创建蛇对象
  function Snake(option) {
    option = option || {};
    this.width = option.width || 20;
    this.height = option.height || 20;
    this.body = [
      {x: 3, y: 2, col: 'green'},//蛇头的位置和颜色
      {x: 2, y: 2, col: 'orange'},//蛇头身体的位置和颜色
      {x: 1, y: 2, col: 'orange'}];
    this.direction = option.direction || 'right';
  }
 
 
  //2 渲染蛇的方法
  Snake.prototype.render = function () {
    // 2-3 为了防止多个sanke渲染到页面上,一渲染之前先清除掉原来的
    for (var i = 0; i < arr.length; i++) {
      map.removeChild(arr[i]);//移除页面上的蛇节
    }
    arr.splice(0,arr.length);//蛇节都被移除掉了,那么数组中也应该都移除.
 
    //2-1 根据body中的个数,动态的创建蛇节
    this.body.forEach(function (item, index) {
      //2-0 动态的创建蛇节
      var snakeNode = document.createElement('div');
      //2-4 遍历添加蛇节新数据
      arr.push(snakeNode);
      snakeNode.style.width = this.width + 'px';
      snakeNode.style.height = this.height + 'px';
      snakeNode.style.position = 'absolute';
      snakeNode.style.left = item.x * this.width + 'px';
      snakeNode.style.top = item.y * this.height + 'px';
      snakeNode.style.backgroundColor = item.col;
      map.appendChild(snakeNode);
 
    }.bind(this))
    //  2-2 上面的this是在snake里面,指向snake.`
    //  否则,默认指向window
  };
 
 
 
  //3 蛇移动的方法:body 头数组赋值给身体.
  Snake.prototype.move = function () {
    //3-1 蛇后面的数据给前面
    for (var i = this.body.length -1; i >0; i--) {
      this.body[i].x = this.body[i - 1].x;
      this.body[i].y = this.body[i - 1].y;
    }
  //  3-2暂时蛇头往右走
  //   this.body[0].x +=1;
  //
  //3-2蛇头一定的位置,要根据蛇的方向来决定
    switch(this.direction){
 
      case 'left':
        this.body[0].x -= 1;
        break;
      case 'right':
        this.body[0].x += 1;
        break;
      case 'top':
        this.body[0].y -= 1;
        break;
      case 'bottom':
        this.body[0].y += 1;
        break;
    }
  };
 
 
 
  //赋予全局变量
  window.Snake = Snake;
})();

--tool.js   常用数据工具封装

//用于存放一些常用的功能性的函数
 
//  function getRandom(){
//
// }
var Tool = {
 //获取min - max之间的随机整数
 getRandom: function(min, max){
  return Math.floor(Math.random() * (max - min + 1)) + min;
 }
}
 
// Tool.getRandom()

以上所述是小编给大家介绍的JavaScript贪吃蛇的实现详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
CSS和JS标签style属性对照表(方便js开发的朋友)
Nov 11 Javascript
jquery监控数据是否变化(修正版)
Apr 12 Javascript
js中 关于undefined和null的区别介绍
Apr 16 Javascript
Jqgrid表格随窗口大小改变而改变的简单实例
Dec 28 Javascript
21个值得收藏的Javascript技巧
Feb 04 Javascript
javascript:void(0)是什么意思及href=#与href=javascriptvoid(0)的区别
Nov 13 Javascript
详解Bootstrap按钮
Jan 04 Javascript
MUI 解决动态列表页图片懒加载再次加载不成功的bug问题
Apr 13 Javascript
详解javascript appendChild()的完整功能
Aug 18 Javascript
js常见遍历操作小结
Jun 06 Javascript
微信小程序实现滑动翻页效果(完整代码)
Dec 06 Javascript
关于Js中new操作符的作用详解
Feb 21 Javascript
小程序绑定用户方案优化小结
May 15 #Javascript
pm2发布node配置文件ecosystem.json详解
May 15 #Javascript
JQuery常见节点操作实例分析
May 15 #jQuery
JS算法题之查找数字在数组中的索引位置
May 15 #Javascript
JQuery属性操作与循环用法示例
May 15 #jQuery
前端js中的事件循环eventloop机制详解
May 15 #Javascript
mpvue小程序循环动画开启暂停的实现方法
May 15 #Javascript
You might like
Ajax+PHP 边学边练之四 表单
2009/11/27 PHP
PHP排序之二维数组的按照字母排序实现代码
2011/08/13 PHP
php中unlink()、mkdir()、rmdir()等方法的使用介绍
2012/12/21 PHP
PHP答题类应用接口实例
2015/02/09 PHP
php计算函数执行时间的方法
2015/03/20 PHP
整理php防注入和XSS攻击通用过滤
2015/09/13 PHP
PHP类的特性实例分析
2016/09/28 PHP
jQuery 第二课 操作包装集元素代码
2010/03/14 Javascript
javascript中使用replaceAll()函数实现字符替换的方法
2010/12/25 Javascript
JS 去前后空格大全(IE9亲测)
2013/07/15 Javascript
js读写cookie实现一个底部广告浮层效果的两种方法
2013/12/29 Javascript
显示今天的日期js代码(阳历和农历)
2014/09/30 Javascript
js实现表单提交后不重新刷新当前页面
2016/11/30 Javascript
vue :src 文件路径错误问题的解决方法
2018/05/15 Javascript
AngularJS标签页tab选项卡切换功能经典实例详解
2018/05/16 Javascript
js/jQuery实现全选效果
2019/06/17 jQuery
微信小程序 组件的外部样式externalClasses使用详解
2019/09/06 Javascript
js 实现碰撞检测的示例
2020/10/28 Javascript
[57:28]2018DOTA2亚洲邀请赛 4.6 淘汰赛 TNC vs Liquid 第一场
2018/04/10 DOTA
Python Tkinter基础控件用法
2014/09/03 Python
Python中使用bidict模块双向字典结构的奇技淫巧
2016/07/12 Python
CentOS 6.5下安装Python 3.5.2(与Python2并存)
2017/06/05 Python
python方向键控制上下左右代码
2018/01/20 Python
Python Flask上下文管理机制实例解析
2020/03/16 Python
python代码实现将列表中重复元素之间的内容全部滤除
2020/05/22 Python
python+appium+yaml移动端自动化测试框架实现详解
2020/11/24 Python
利用css3如何设置没有上下边的列表间隔线
2017/07/03 HTML / CSS
美国标志性加大尺码时装品牌:Ashley Stewart
2016/12/15 全球购物
给老师的道歉信
2014/01/11 职场文书
公司授权委托书
2014/04/04 职场文书
大学生党员自我剖析材料
2014/10/06 职场文书
教师辞职信范文
2015/02/28 职场文书
2015年暑假生活总结
2015/07/13 职场文书
婚宴祝酒词大全
2015/08/10 职场文书
关于环保的广播稿
2015/12/17 职场文书
vscode远程免密登入Linux服务器的配置方法
2022/06/28 Servers