JavaScript Canvas 游戏:使用类管理多个敌人实例的教程

javascript canvas 游戏:使用类管理多个敌人实例的教程

在JavaScript Canvas游戏中,当需要管理多个独立移动的敌人或其他游戏实体时,直接使用全局变量会导致所有实体共享相同的状态,从而表现出同步且非预期的行为。本文将深入探讨这一常见问题,并提供一个基于JavaScript类的面向对象解决方案,通过为每个实体创建独立实例来有效管理其各自的位置、速度和行为,确保每个敌人都能独立运动并响应环境,从而构建出更复杂和动态的游戏场景。

问题描述:全局状态与多实体管理

在开发基于JavaScript Canvas的游戏时,一个常见的挑战是如何有效地管理多个游戏实体,例如多个敌人。如果每个敌人的位置、速度等属性都依赖于全局变量,那么当游戏中出现多个敌人时,它们将无法独立行动。

考虑以下初始代码片段,它尝试绘制并移动一个敌人:

var canvas = document.getElementById("canvas");var ctx = canvas.getContext("2d");var x = 0; // 全局X坐标var y = 0; // 全局Y坐标var x_add = 2; // 全局X方向速度var y_add = 2; // 全局Y方向速度function animate(){    draw();    setTimeout(animate, 10);};function draw(){    ctx.clearRect(0,0,1500, 500);    draw_enemy(900, 100, "red", 40, 50); // 绘制一个敌人};function draw_enemy(start_x, start_y, fill, w, h){    // 边界检测和速度反转逻辑    if(x + w + start_x >= 1000){ // 注意:这里使用了硬编码的画布宽度        x_add = -2;    }    if(y + h + start_y >= 500){ // 注意:这里使用了硬编码的画布高度        y_add = -2;    }    if(y  + start_y <= 0){        y_add = 2;    }    if(x + start_x <= 0){        x_add = 2;    }    x += x_add; // 更新全局X坐标    y += y_add; // 更新全局Y坐标    ctx.fillStyle = fill;    ctx.fillRect(x + start_x, y + start_y, w, h);};animate();

以及对应的HTML结构:

            local storage test                                

这段代码的问题在于,x、y、x_add 和 y_add 都是全局变量。当只绘制一个敌人时,一切正常。然而,如果尝试绘制两个或更多敌人,例如在 draw 函数中调用 draw_enemy 两次,所有敌人都会共享这些全局变量。这意味着:

立即学习“Java免费学习笔记(深入)”;

它们将使用相同的 x 和 y 坐标,导致它们在屏幕上重叠或表现出相同的位移。当任何一个敌人触碰到边界时,它会修改全局的 x_add 或 y_add,从而影响所有其他敌人的移动方向,使得它们的行为变得同步且随机。

这种共享全局状态的方式使得每个敌人都无法拥有独立的运动轨迹和行为逻辑,这显然不符合游戏设计的需求。

解决方案:使用JavaScript类进行面向对象管理

为了解决上述问题,最佳实践是采用面向对象编程的思想,使用JavaScript的 class 语法来定义敌人的蓝图。通过类,我们可以创建多个独立的敌人实例,每个实例都拥有自己的属性(如位置、速度、颜色、大小等)和方法(如绘制、更新)。

1. 定义 Enemy 类

首先,我们创建一个 Enemy 类,它将封装每个敌人的所有相关属性和行为。

class Enemy {  constructor(color, initialX, initialY, width, height, initialVx, initialVy) {    // 为每个敌人实例初始化独立的属性    this.x = initialX !== undefined ? initialX : 50 + Math.random() * (canvas.width - 100);    this.y = initialY !== undefined ? initialY : 50 + Math.random() * (canvas.height - 100);    this.w = width !== undefined ? width : 40;    this.h = height !== undefined ? height : 50;    this.c = color !== undefined ? color : 'red'; // 颜色    this.vx = initialVx !== undefined ? initialVx : 2; // X方向速度    this.vy = initialVy !== undefined ? initialVy : 2; // Y方向速度  }  // 绘制敌人的方法  draw() {    ctx.fillStyle = this.c;    ctx.fillRect(this.x, this.y, this.w, this.h);  }  // 更新敌人状态(位置和边界检测)的方法  update() {    // 边界检测:使用this.x, this.y, this.w, this.h确保是当前实例的属性    if (this.x + this.w >= canvas.width || this.x = canvas.width ? canvas.width - this.w : 0;    }    if (this.y + this.h >= canvas.height || this.y = canvas.height ? canvas.height - this.h : 0;    }    // 更新位置:使用this.vx, this.vy确保是当前实例的速度    this.x += this.vx;    this.y += this.vy;    this.draw(); // 更新后立即绘制  }}

在 Enemy 类中:

constructor 方法用于初始化每个敌人实例的属性,如 x、y(位置)、w、h(宽度、高度)、c(颜色)以及 vx、vy(X、Y方向的速度)。通过 this 关键字,确保这些属性是每个实例独有的。draw() 方法负责根据当前实例的 x、y、w、h 和 c 属性在 Canvas 上绘制敌人。update() 方法负责更新敌人实例的位置,并执行边界检测。同样,它使用 this 关键字来访问和修改当前实例的属性。

2. 管理多个敌人实例

为了管理多个敌人,我们需要一个数组来存储 Enemy 类的实例。

var canvas = document.getElementById("canvas");var ctx = canvas.getContext("2d");let enemies = []; // 存储所有敌人实例的数组// 创建多个敌人实例并添加到数组中function createEnemies(count = 5) {  for (let i = 0; i < count; i++) {    // 可以传入不同的参数来创建具有不同初始状态的敌人    enemies.push(new Enemy('red'));   }}createEnemies(3); // 创建3个红色敌人enemies.push(new Enemy('green', 100, 200, 50, 50, 3, -3)); // 创建一个绿色敌人,并指定其初始状态enemies.push(new Enemy('blue', 500, 150, 60, 40, -2, 4)); // 创建一个蓝色敌人

3. 改进游戏循环

现在,游戏的主循环(animate 和 draw 函数)需要进行修改,以便遍历 enemies 数组中的所有敌人实例,并分别调用它们的 update 方法。

function draw() {  ctx.clearRect(0, 0, canvas.width, canvas.height); // 清除整个画布  // 遍历敌人数组,对每个敌人实例调用其update方法  enemies.forEach(enemy => enemy.update());}function animate() {  draw();  // 推荐使用 requestAnimationFrame() 替代 setTimeout(),以获得更流畅的动画  // requestAnimationFrame(animate);  setTimeout(animate, 10); }animate(); // 启动动画

完整示例代码

HTML (index.html):

            JavaScript Canvas 游戏:多敌人管理                    body { margin: 0; overflow: hidden; display: flex; justify-content: center; align-items: center; min-height: 100vh; background-color: #f0f0f0; }            canvas { border: 1px solid black; background-color: #eee; }                                    

JavaScript (script.js):

var canvas = document.getElementById("canvas");var ctx = canvas.getContext("2d");let enemies = []; // 存储所有敌人实例的数组class Enemy {  constructor(color, initialX, initialY, width, height, initialVx, initialVy) {    this.w = width !== undefined ? width : 40;    this.h = height !== undefined ? height : 50;    this.x = initialX !== undefined ? initialX : Math.random() * (canvas.width - this.w);    this.y = initialY !== undefined ? initialY : Math.random() * (canvas.height - this.h);    this.c = color !== undefined ? color : 'red';    this.vx = initialVx !== undefined ? initialVx : 2 * (Math.random() > 0.5 ? 1 : -1); // 随机初始速度方向    this.vy = initialVy !== undefined ? initialVy : 2 * (Math.random() > 0.5 ? 1 : -1);  }  draw() {    ctx.fillStyle = this.c;    ctx.fillRect(this.x, this.y, this.w, this.h);  }  update() {    // 边界检测和速度反转    if (this.x + this.w >= canvas.width) {      this.x = canvas.width - this.w; // 修正位置      this.vx *= -1;    }    if (this.x = canvas.height) {      this.y = canvas.height - this.h; // 修正位置      this.vy *= -1;    }    if (this.y <= 0) {      this.y = 0; // 修正位置      this.vy *= -1;    }    this.x += this.vx;    this.y += this.vy;    this.draw();  }}// 创建多个敌人实例function createEnemies(count = 5) {  for (let i = 0; i  enemy.update()); // 遍历并更新所有敌人}function animate() {  draw();  // 推荐使用 requestAnimationFrame() 替代 setTimeout() 以获得更流畅的动画  requestAnimationFrame(animate);   // setTimeout(animate, 10); }animate();

注意事项与最佳实践

使用 canvas.width 和 canvas.height: 在进行边界检测时,始终使用 canvas.width 和 canvas.height 而不是硬编码的数值。这使得你的游戏能够更好地适应不同大小的 Canvas,增加代码的灵活性和可维护性。requestAnimationFrame() vs. setTimeout(): 对于游戏循环,强烈推荐使用 requestAnimationFrame() 而非 setTimeout()。requestAnimationFrame() 会在浏览器下一次重绘之前调用指定的回调函数,它与浏览器的刷新率同步,能够提供更平滑、更高效的动画,并节省CPU资源(当页面在后台时会暂停)。构造函数的灵活性: 在 Enemy 类的 constructor 中,你可以根据需要传入更多参数,例如不同的颜色、初始位置、生命值、防御力等。这使得创建各种具有独特属性的敌人变得非常容易。数组迭代方法: forEach 是一种简洁且常用的数组迭代方法,它比传统的 for 循环更具可读性。在处理游戏实体数组时,它是一个很好的选择。碰撞修正: 在边界检测中,除了反转速度外,最好也修正一下对象的位置,将其精确地放置在边界上,以避免在某些帧率下对象“卡”在边界内或穿透边界的问题。示例代码中已加入了简单的位置修正。

总结

通过采用面向对象的设计方法,利用JavaScript的 class 语法,我们可以为游戏中的每个实体创建独立的实例。每个实例都拥有自己的状态和行为,并通过一个数组进行统一管理。这种方法不仅解决了多实体共享全局状态的问题,使得每个敌人都能独立运动,还大大提高了代码的模块化、可读性和可维护性,为构建更复杂、更动态的Canvas游戏奠定了坚实的基础。

以上就是JavaScript Canvas 游戏:使用类管理多个敌人实例的教程的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1582579.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月22日 23:17:39
下一篇 2025年12月22日 23:17:51

相关推荐

  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 为什么设置 `overflow: hidden` 会导致 `inline-block` 元素错位?

    overflow 导致 inline-block 元素错位解析 当多个 inline-block 元素并列排列时,可能会出现错位显示的问题。这通常是由于其中一个元素设置了 overflow 属性引起的。 问题现象 在不设置 overflow 属性时,元素按预期显示在同一水平线上: 不设置 overf…

    2025年12月24日 好文分享
    400
  • 微信小程序文本省略后如何避免背景色溢出?

    去掉单行文本溢出多余背景色 在编写微信小程序时,如果希望文本超出宽度后省略显示并在末尾显示省略号,但同时还需要文本带有背景色,可能会遇到如下问题:文本末尾出现多余的背景色块。这是因为文本本身超出部分被省略并用省略号代替,但其背景色依然存在。 要解决这个问题,可以采用以下方法: 给 text 元素添加…

    2025年12月24日
    000
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • Flex 布局左右同高怎么实现?

    flex布局左右同高 在flex布局中,左右布局的元素高度不一致时,想要让边框延伸到最大高度,可以采用以下方法: 基于当前结构的方法: 给.rht和.lft盒子添加: .rht { height: min-content;} 这样可以使弹性盒子被子盒子内容撑开。 使用javascript获取.rht…

    2025年12月24日
    000
  • inline-block元素错位了,是为什么?

    inline-block元素错位背后的原因 inline-block元素是一种特殊类型的块级元素,它可以与其他元素行内排列。但是,在某些情况下,inline-block元素可能会出现错位显示的问题。 错位的原因 当inline-block元素设置了overflow:hidden属性时,它会影响元素的…

    2025年12月24日
    000
  • 为什么使用 inline-block 元素时会错位?

    inline-block 元素错位成因剖析 在使用 inline-block 元素时,可能会遇到它们错位显示的问题。如代码 demo 所示,当设置了 overflow 属性时,a 标签就会错位下沉,而未设置时却不会。 问题根源: overflow:hidden 属性影响了 inline-block …

    2025年12月24日
    000
  • 如何去除带有背景色的文本单行溢出时的多余背景色?

    带背景色的文字单行溢出处理:去除多余的背景色 当一个带有背景色的文本因单行溢出而被省略时,可能会出现最后一个背景色块多余的情况。针对这种情况,可以通过以下方式进行处理: 在示例代码中,问题在于当文本溢出时,overflow: hidden 属性会导致所有文本元素(包括最后一个)都隐藏。为了解决该问题…

    2025年12月24日
    000
  • 如何解决 CSS 中文本溢出时背景色也溢出的问题?

    文字单行溢出省略号时,去掉多余背景色的方法 在使用 css 中的 text-overflow: ellipsis 属性时,如果文本内容过长导致一行溢出,且文本带有背景色,溢出的部分也会保留背景色。但如果想要去掉最后多余的背景色,可以采用以下方法: 给 text 元素添加一个 display: inl…

    2025年12月24日
    200
  • 如何用CSS实现文本自动展开,并在超出两行后显示展开下箭头?

    CSS实现文本自动展开的难题 一段文本超出两行后自动溢出的效果,需要添加一个展开下箭头指示用户有隐藏内容。实现这一需求时,面临以下难题: 判断是否超过两行溢出取消省略号,用展开下箭头代替 解决思路:参考大佬文章 这个问题的解决方法,可以参考本站大佬的文章CSS 实现多行文本“展开收起”,该文章正是针…

    2025年12月24日
    000
  • 如何去除单行溢出文本中的冗余背景色?

    带背景色的文字单行溢出省略号,如何去除冗余背景色? 在使用 css 样式时,为单行溢出文本添加背景色可能会导致最后一行文本中的冗余背景色。为了解决这个问题,可以为文本元素添加额外的 css 样式: text { display: inline-block;} 添加这个样式后,文字截断将基于文本块进行…

    2025年12月24日
    000
  • 如何用 CSS 实现纵向文字溢出省略号?

    纵向文字溢出的省略号处理方案 对于纵向展示的文字,传统的横向溢出省略方案(使用 overflow: hidden; text-overflow: ellipsis;)不适用。若需在纵向展示时实现省略号,可考虑以下 css 解决方案: 垂直排版 通过将文字排版模式改为垂直,可以解决纵向溢出的问题。使用…

    2025年12月24日
    000
  • 前端代码辅助工具:如何选择最可靠的AI工具?

    前端代码辅助工具:可靠性探讨 对于前端工程师来说,在HTML、CSS和JavaScript开发中借助AI工具是司空见惯的事情。然而,并非所有工具都能提供同等的可靠性。 个性化需求 关于哪个AI工具最可靠,这个问题没有一刀切的答案。每个人的使用习惯和项目需求各不相同。以下是一些影响选择的重要因素: 立…

    2025年12月24日
    000
  • 图片轮播效果实现的最佳方案是什么?

    实现图片切换效果的妙招 在浏览网站时,你可能会遇到引人注目的图片轮播效果,想要尝试自己实现。然而,实现效果可能并不令人满意,想知道问题的根源吗? 问题在于你使用的是 标签,直接改变图片位置,这会导致图像质量降低。更好的办法是使用 元素并使用 css background-image 属性,同时改变 …

    2025年12月24日
    000
  • 动画滚动表格时,如何防止表格内容超出表头继续滚动?

    动画滚动效果时表格内容超出表头 你给出了一个带有自动滚动的表格,但发现表格中的行在超过表头时仍然会继续滚动。要解决这个问题,需要对你的 css 代码进行一些调整。 以下是解决你问题的 css 代码: @keyframes table { 0% { transform: translateY(0); …

    2025年12月24日
    000
  • 图片轮播效果实现问题:使用 transform: translateX 实现图片切换,为何效果不理想?

    图片切换效果实现 问题: 本想实现一个常见的图片轮播效果,却多次碰壁,请指教问题所在。 效果展示: 原样式自实现效果 代码: .slider { width: 700px; height: 400px; overflow: hidden; position: relative; } .slider-…

    2025年12月24日 好文分享
    000
  • 表格自动滚动时,tbody溢出表头怎么办?

    表格自动滚动时,tbody溢出表头? 当使用动画实现表格自动滚动时,通常需要确保tbody的内容在滚动过程中不会超出表头。但是,在遇到tbody内容超过表头滚动的问题时,可以考虑以下解决方法: 在代码中定位table的样式,添加overflow: hidden;属性。这将隐藏超出table范围的子元…

    2025年12月24日
    000
  • 布局 – CSS 挑战

    您可以在 github 仓库中找到这篇文章中的所有代码。 您可以在这里查看视觉效果: 固定导航 – 布局 – codesandbox两列 – 布局 – codesandbox三列 – 布局 – codesandbox圣杯 &#8…

    2025年12月24日
    000
  • 表格主体滚动时,为何超出表头消失?

    在表中实现自动滚动时,body总是超过表头消失的原因 当为表格主体(tbody)设置了动画滚动时,tbody会沿着纵轴移动,当tbody完全滚动出表格(table)的范围时,tbody就会从视图中消失。然而,在给出的代码中,没有对表格本身或表头(thead)设置任何限制,导致tbody在滚动出表格范…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信