
在开发基于j*ascript canvas的2d游戏时,管理屏幕上的多个动态实体(例如敌人、子弹或道具)是一个核心挑战。初学者常常会遇到一个常见问题:当尝试绘制和更新多个同类型实体时,它们却表现出完全相同的行为,或者它们的运动状态相互干扰,导致游戏逻辑混乱。本文将深入探讨这个问题的原因,并提供一个基于j*ascript类的优雅解决方案。
在早期的游戏开发尝试中,开发者可能会为敌人创建一个函数,并使用全局变量来控制其位置和速度。例如:
var x = 0;
var y = 0;
var x_add = 2; // 全局X轴速度
var y_add = 2; // 全局Y轴速度
function draw_enemy(start_x, start_y, fill, w, h){
// 边界检测和速度反转
if(x + w + start_x >= 1000){ // 假设canvas宽度为1000
x_add = -2;
}
if(y + h + start_y >= 500){ // 假设canvas高度为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);
};当只有一个敌人调用 draw_enemy 函数时,一切看起来正常。然而,一旦尝试绘制第二个敌人,问题就暴露无遗:所有的敌人都会根据全局变量 x、y、x_add 和 y_add 进行移动。这意味着,当第一个敌人触碰到边界并改变了 x_add 的值时,紧接着绘制的第二个敌人也会使用这个被修改过的 x_add 值,导致所有敌人步调一致,失去独立性。问题的核心在于,每个敌人实例都需要拥有自己独立的坐标和速度状态,而不是共享全局变量。
解决上述问题的最佳方法是使用J*aScript的类(Class)。类允许我们定义一个蓝图,通过这个蓝图可以创建多个具有独立属性和行为的对象实例。每个实例都拥有自己的状态(如位置、速度、颜色等),从而实现独立的运动和交互。
首先,我们定义一个 Enemy 类,它将包含每个敌人所需的属性和方法。
// 获取Canvas上下文
var canvas = document.getElementById("canvas");
var ctx = canvas.getContext("2d");
// 用于存储所有敌人实例的数组
let enemies = [];
class Enemy {
/**
* 构造函数,用于初始化每个敌人的属性
* @param {string} color - 敌人的颜色
* @param {number} initialX - 敌人初始X坐标 (可选)
* @param {number} initialY - 敌人初始Y坐标 (可选)
*/
constructor(color, initialX = null, initialY = null) {
// 随机或指定初始位置
this.x = initialX !== null ? initialX : 50 + Math.random() * (canvas.width - 100);
this.y = initialY !== null ? initialY : 50 + Math.random() * (canvas.height - 100);
this.w = 40; // 宽度
this.h = 50; // 高度
this.color = color; // 颜色
this.vx = 2; // X轴速度
this.vy = 2; // Y轴速度
}
/**
* 绘制敌人到Canvas上
*/
draw() {
ctx.fillStyle = this.color;
ctx.fillRect(this.x, this.y, this.w, this.h);
}
/**
* 更新敌人的位置和处理边界碰撞
*/
update() {
// 边界检测和速度反转
if (this.x + this.w >= canvas.width) {
this.vx = -Math.abs(this.vx); // 确保速度为负
}
if (this.y + this.h >= canvas.height) {
this.vy = -Math.abs(this.vy); // 确保速度为负
}
if (this.y <= 0) {
this.vy = Math.abs(this.vy); // 确保速度为正
}
if (this.x <= 0) {
this.vx = Math.abs(this.vx); // 确保速度为正
}
// 更新位置
this.x += this.vx;
this.y += this.vy;
// 绘制更新后的敌人
this.draw();
}
}在 Enemy 类中:
Facetune
一款在线照片和视频编辑工具,允许用户创建AI头像
109
查看详情
接下来,我们需要创建 Enemy 类的实例,并将它们存储在一个数组中,以便在游戏循环中统一管理。
// 创建多个敌人实例
function createEnemies(count = 5) {
for (let i = 0; i < count; i++) {
enemies.push(new Enemy('red')); // 创建5个红色敌人
}
}
createEnemies();
// 也可以单独添加特定颜色或位置的敌人
enemies.push(new Enemy('green', 100, 200)); // 一个绿色敌人,初始位置(100, 200)
enemies.push(new Enemy('blue', 500, 150)); // 一个蓝色敌人,初始位置(500, 150)在游戏的主循环中,我们需要清空Canvas,然后遍历 enemies 数组,对每个敌人实例调用其 update() 方法。
/**
* 游戏的每一帧绘制函数
*/
function drawGameFrame() {
// 清空整个Canvas,为下一帧做准备
ctx.clearRect(0, 0, canvas.width, canvas.height);
// 遍历所有敌人,更新并绘制它们
enemies.forEach(enemy => enemy.update());
// 等同于:
// for (let i = 0; i < enemies.length; i++) {
// enemies[i].update();
// }
}
/**
* 动画循环函数
*/
function animate() {
drawGameFrame();
// 建议使用 requestAnimationFrame() 替代 setTimeout() 以获得更平滑的动画
// requestAnimationFrame(animate);
setTimeout(animate, 10); // 每10毫秒更新一次
}
// 启动动画
animate();var canvas = document.getElementById("canvas");
var ctx = canvas.getContext("2d");
let enemies = [];
class Enemy {
constructor(color, initialX = null, initialY = null) {
this.x = initialX !== null ? initialX : 50 + Math.random() * (canvas.width - 100);
this.y = initialY !== null ? initialY : 50 + Math.random() * (canvas.height - 100);
this.w = 40;
this.h = 50;
this.color = color;
this.vx = 2;
this.vy = 2;
}
draw() {
ctx.fillStyle = this.color;
ctx.fillRect(this.x, this.y, this.w, this.h);
}
update() {
if (this.x + this.w >= canvas.width) {
this.vx = -Math.abs(this.vx);
}
if (this.y + this.h >= canvas.height) {
this.vy = -Math.abs(this.vy);
}
if (this.y <= 0) {
this.vy = Math.abs(this.vy);
}
if (this.x <= 0) {
this.vx = Math.abs(this.vx);
}
this.x += this.vx;
this.y += this.vy;
this.draw();
}
}
function createEnemies(count = 5) {
for (let i = 0; i < count; i++) {
enemies.push(new Enemy('red'));
}
}
createEnemies();
enemies.push(new Enemy('green', 100, 200));
enemies.push(new Enemy('blue', 500, 150));
function drawGameFrame() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
enemies.forEach(enemy => enemy.update());
}
function animate() {
drawGameFrame();
// 建议使用 requestAnimationFrame() 替代 setTimeout()
// requestAnimationFrame(animate);
setTimeout(animate, 10);
}
animate();<!DOCTYPE html>
<html>
<head>
<title>J*aScript Canvas 敌人管理</title>
<style>
body { margin: 0; display: flex; justify-content: center; align-items: center; min-height: 100vh; background-color: #f0f0f0; }
canvas { border: 1px solid black; background-color: white; }
</style>
</head>
<body>
<canvas id="canvas" width="1000" height="500"></canvas>
<script src="script.js"></script>
</body>
</html>通过采用J*aScript类来封装游戏实体,我们成功地解决了多个实体共享全局状态导致行为一致的问题。每个 Enemy 实例都拥有独立的 x, y, vx, vy 等属性,并通过各自的 update() 方法独立地更新状态和处理逻辑。结合数组来管理这些实例,并在游戏主循环中遍历更新,不仅使得代码结构清晰、逻辑独立,还大大提升了游戏的可扩展性和可维护性,为构建更复杂、更动态的Canvas游戏奠定了坚实的基础。
以上就是J*aScript Canvas游戏:高效管理多个敌人实体教程的详细内容,更多请关注其它相关文章!
# 遍历
# 四月营销推广文案范文怎么写
# 濮阳 网站建设
# 乳胶床垫如何做营销推广
# 磐石集团网站建设
# 大理网站建设公司
# 教人如何seo的软文
# 麻将游戏推广网站
# 排名页面与关键词不对号
# 西青区网站推广营销公司
# 房地产网站推广入门书籍
# 清空
# 人与
# 可选
# 第二个
# 而不是
# javascript
# 全局变量
# 回调
# 多个
# r
# canva
# 重绘
# 常见问题
# 游戏开发
# win
# 回调函数
# 浏览器
# 编码
# js
# html
# java
相关栏目:
【
Google疑问12 】
【
Facebook疑问10 】
【
优化推广96088 】
【
技术知识133117 】
【
IDC资讯59369 】
【
网络运营7196 】
【
IT资讯61894 】
相关推荐:
漫蛙漫画官方版直通入口 2025漫蛙漫画免注册访问说明
招商淘客入门指南
如何解决Casbin日志与应用日志不统一的问题,使用casbin/psr3-bridge实现无缝集成
大熊猫抓取竹子的“大拇指”其实是什么?蚂蚁庄园课堂今天答案最新11月30日
智慧职教mooc平台登录网址 智慧职教mooc官网直达
如何高效地基于键列值映射DataFrame中的多个列
2025SNH48年度青春盛典门票价格及购买方式
《知到》打卡课程方法
解决Flex容器横向滚动内容截断与偏移问题
海棠阅读登录教程_详细讲解海棠登录操作
使用document.execCommand实现Web文本编辑器加粗/取消加粗
Win10显卡驱动安装失败怎么办 Win10使用DDU彻底卸载驱动【解决】
Animex动漫社社登录官网 Animex动漫社资源社入口直达
泰拉瑞亚水晶无法放置问题
C++ switch case字符串_C++如何实现字符串switch匹配
《小黑盒》删除历史浏览方法
顺丰快递在线查询系统 顺丰快递官方查单入口
基于键值条件高效映射 Pandas DataFrame 多列数据
win11如何诊断DirectX问题 Win11运行dxdiag工具排查显卡故障【排错】
TikTok收藏夹无法删除视频如何解决 TikTok收藏管理优化方法
教资成绩怎么查询
MySQL多重JOIN技巧:高效关联同一表获取多角色信息
mysql通配符能用于日志查询吗_mysql通配符在系统日志查询中的实际使用方法
高德地图导航路线偏差报警频繁怎么办 高德地图路线偏差修复与优化方法
德邦快递查询入口登录官网 德邦快递单号查询系统入口
如何定制PrimeNG Sidebar的背景颜色
邦丰播放器频道搜索设置
在React中正确处理HTML input type="number"的数值类型
《oppo商城》维修服务位置
抖音号已注销怎么解绑企业认证?不解绑企业认证会怎样?
全球各国上班时间表外贸邮件时间
b站如何管理订阅_b站订阅标签分类管理
263企业邮箱如何设置邮件转发功能
J*aScript:从子元素中批量移除特定CSS类
在J*a中如何实现类的继承与方法重用_OOP继承方法重用技巧分享
微星主板BIOS怎么调整内存时序_内存参数手动优化BIOS设置教程
在Spring Boot Thymeleaf中利用布尔属性实现容器的条件显示
Django模型动态关联检查:高效管理复杂关系
实现二叉树的层序插入:基于树大小的路径导航
diskgenius分区工具如何设置Bios启动项
mysql数据库索引类型有哪些_mysql索引类型解析
在VS Code中进行数据科学和机器学习开发
Lar*el怎么实现全文搜索_Lar*el Scout集成Algolia教程
《华夏千秋》龙女试炼功法获取方法
三星A55应用闪退排查步骤_Samsung A55稳定性优化技巧
在XML中嵌入二进制数据(如图片)的最佳实践是什么? Base64编码与解析注意事项
如何在mysql中比较InnoDB和MyISAM区别
j*a中赋值运算符是什么?
惠普电脑BIOS界面看不懂怎么办_HP电脑BIOS功能选项解读与设置
如何外贸网站设计-能留住客户提升用户体验!
2025-10-09
运城市盐湖区信雨科技有限公司是一家深耕海外推广领域十年的专业服务商,作为谷歌推广与Facebook广告全球合作伙伴,聚焦外贸企业出海痛点,以数字化营销为核心,提供一站式海外营销解决方案。公司凭借十年行业沉淀与平台官方资源加持,打破传统外贸获客壁垒,助力企业高效开拓全球市场,成为中小企业出海的可靠合作伙伴。