0

0

HTML 赛车游戏:精准实现速度递增与得分判定逻辑

霞舞

霞舞

发布时间:2026-01-12 14:12:39

|

838人浏览过

|

来源于php中文网

原创

HTML 赛车游戏:精准实现速度递增与得分判定逻辑

本文详解如何修复 html 赛车游戏中“每5秒增速+1(起始为5)”和“仅当玩家车尾完全超越敌车车尾时加10分”两大核心逻辑,解决重启异常、误判得分、速度失控等常见 bug。

在开发基于 HTML/CSS/JavaScript 的横向卷轴赛车游戏时,速度动态增长精准得分判定是提升游戏挑战性与公平性的关键。原始代码存在多个严重逻辑缺陷:player.speed 在 initializeGame() 中未重置为初始值 5;player.score++ 被错误地置于 runGame() 主循环末尾,导致帧率依赖型无效累加;更关键的是,得分条件完全缺失——未通过 DOM 元素边界比对实现“车尾超越”判断,而是用无意义的全局自增替代。

✅ 正确实现速度递增(每5秒 +1,起始为5)

需引入时间戳追踪机制,避免使用 setInterval(易因帧率波动或多次初始化导致计时错乱)。在 runGame() 中记录上一次增速时间,并在每次执行时检查是否已过 5000ms:

let lastSpeedIncreaseTime = 0;

function updateSpeed() {
  const now = Date.now();
  if (now - lastSpeedIncreaseTime >= 5000) {
    player.speed++;
    lastSpeedIncreaseTime = now;
  }
}

并在 runGame() 的主逻辑中调用:

if (player.start) {
  updateSpeed(); // ← 新增:精准控制增速节奏
  moveLines();
  moveEnemy(car);
  // ...其余移动逻辑
}

同时,必须在 initializeGame() 中显式重置所有状态

立即学习前端免费学习笔记(深入)”;

ReRoom AI
ReRoom AI

专为室内设计打造的AI渲染工具,可以将模型图、平面图、草图、照片转换为高质量设计效果图。

下载
function initializeGame() {
  startScreen.classList.add('hide');
  gameArea.innerHTML = "";

  player.start = true;
  player.score = 0;
  player.speed = 5; // ✅ 强制重置起始速度
  player.x = 0;
  player.y = 0;
  lastSpeedIncreaseTime = Date.now(); // ✅ 重置计时起点

  // ...创建道路线、玩家车、敌车等
}

✅ 精准实现“车尾超越”得分逻辑(+10 分/次)

原始代码中 player.score++ 是纯计数器,与游戏物理完全脱钩。正确做法是:每一帧检测每个敌车是否满足 myCar.bottom ≤ enemyCar.bottom → myCar 已完全越过敌车,且该敌车此前未被计分

为此需为每个敌车元素添加自定义属性标记(如 data-scored="false"),并在 moveEnemy() 中进行判定:

function moveEnemy(myCar) {
  const myRect = myCar.getBoundingClientRect();
  const myCarBottom = myRect.bottom;

  document.querySelectorAll('.enemyCar').forEach(enemyCar => {
    const enemyRect = enemyCar.getBoundingClientRect();
    const enemyCarBottom = enemyRect.bottom;

    // ✅ 核心条件:玩家车尾 ≤ 敌车车尾 → 完全超越
    if (myCarBottom <= enemyCarBottom && !enemyCar.dataset.scored) {
      player.score += 10;
      enemyCar.dataset.scored = "true"; // 防止重复计分
      score.innerText = `Score: ${player.score}\nSpeed: ${player.speed}`;
    }

    // 敌车移出屏幕后重置(并清除计分标记)
    if (enemyRect.top > window.innerHeight) {
      enemyCar.style.top = "-100px";
      enemyCar.style.left = Math.floor(Math.random() * 350) + "px";
      enemyCar.dataset.scored = "false"; // 重置可计分状态
    }

    // 更新敌车位置(按当前 player.speed)
    enemyCar.style.top = (parseFloat(enemyCar.style.top) || 0) + player.speed + "px";
  });
}
⚠️ 注意:getBoundingClientRect() 返回的是视口坐标,确保 无意外滚动偏移;若游戏区域有 transform 或 position: fixed,需统一使用 offsetTop + clientHeight 计算相对位置。

? 其他关键修复点

  • 移除冗余 player.score++:原始代码中该行会导致每帧强制 +1,彻底删除;
  • 修正 requestAnimationFrame 调用位置:应在 runGame() 末尾递归调用,而非在 initializeGame() 中单次调用(否则重启后动画停止);
  • 防止多次初始化冲突:为 startScreen.addEventListener 添加 .once('click') 或在 initializeGame() 开头检查 player.start 状态;
  • 优化性能:querySelectorAll 和 getBoundingClientRect() 开销较大,可缓存元素引用或使用 offsetTop/offsetHeight 替代(若布局固定)。

✅ 最终效果验证清单

问题 是否修复 验证方式
起始速度恒为 5 重启后 console.log(player.speed) 输出 5
每 5 秒准确 +1 启动后等待 5s,观察 speed 变为 6;再等 5s 变为 7
得分仅触发于“车尾超越” 敌车从下方接近时,仅当黄色车完全盖过绿色车底部才 +10
重启不累积计时器 连续重启 5 次,每次均从 5 开始,5s 后变 6
无重复/漏计分 敌车反复刷出,每次仅首次超越计分

通过以上结构化重构,游戏逻辑从“视觉动画演示”升级为具备严谨物理判定与可控成长曲线的完整小游戏。核心在于:用时间戳替代 setInterval 实现稳定节奏,用 DOM 边界比对替代臆测条件实现精准事件触发——这是前端游戏开发中复用性与健壮性的基石实践。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

552

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

374

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

730

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

475

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

394

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

990

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

656

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

551

2023.09.20

Java 项目构建与依赖管理(Maven / Gradle)
Java 项目构建与依赖管理(Maven / Gradle)

本专题系统讲解 Java 项目构建与依赖管理的完整体系,重点覆盖 Maven 与 Gradle 的核心概念、项目生命周期、依赖冲突解决、多模块项目管理、构建加速与版本发布规范。通过真实项目结构示例,帮助学习者掌握 从零搭建、维护到发布 Java 工程的标准化流程,提升在实际团队开发中的工程能力与协作效率。

6

2026.01.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.9万人学习

CSS教程
CSS教程

共754课时 | 18.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号