0

0

修复 Rock-Paper-Scissors 屏保中计数器失效的关键问题

聖光之護

聖光之護

发布时间:2026-03-16 09:06:17

|

792人浏览过

|

来源于php中文网

原创

修复 Rock-Paper-Scissors 屏保中计数器失效的关键问题

本文详解为何 updatecounters() 无法实时反映 emoji 类型变化,并提供两种专业级解决方案:同步更新 dom 元素的 css 类名,或改用状态变量集中管理计数逻辑。

本文详解为何 updatecounters() 无法实时反映 emoji 类型变化,并提供两种专业级解决方案:同步更新 dom 元素的 css 类名,或改用状态变量集中管理计数逻辑。

在你的 Rock-Paper-Scissors 屏保实现中,updateCounters() 函数本身逻辑正确——它通过 document.querySelectorAll('.scissors') 等选择器统计对应类名的元素数量,并更新 UI。但问题根源在于:当碰撞发生、emoji 文本内容(textContent)被修改后,其 CSS 类名(.rock / .paper / .scissors)并未同步更新

例如,当一个 ?(paper,带 .paper 类)与 ✂️(scissors,带 .scissors 类)碰撞时,你的 updateState() 正确地将 paper 的文本改为 ✂️,但该元素仍保留 .paper 类。而 updateCounters() 仅依赖类名计数,因此它仍把该元素计入 paperCount,而非 scissorsCount —— 导致计数器始终“卡”在初始值,无法响应动态变化。

✅ 方案一:同步更新 class 名(推荐,语义清晰)

在 updateState() 中,每次修改 textContent 时,必须同步增删 CSS 类:

function updateState(emoji1, emoji2) {
  const isScissors = (el) => el.textContent === '✂️';
  const isRock     = (el) => el.textContent === '?'; // 注意:原文中使用了 ?,实际应为 ?(rock)和 ?(paper)
  const isPaper    = (el) => el.textContent === '?';

  if (isScissors(emoji1) && isPaper(emoji2)) {
    emoji2.textContent = '✂️';
    emoji2.classList.remove('paper');   // 移除旧类
    emoji2.classList.add('scissors');   // 添加新类
  } else if (isPaper(emoji1) && isScissors(emoji2)) {
    emoji1.textContent = '✂️';
    emoji1.classList.remove('paper');
    emoji1.classList.add('scissors');
  } else if (isRock(emoji1) && isScissors(emoji2)) {
    emoji2.textContent = '?';
    emoji2.classList.remove('scissors');
    emoji2.classList.add('rock');
  } else if (isScissors(emoji1) && isRock(emoji2)) {
    emoji1.textContent = '?';
    emoji1.classList.remove('scissors');
    emoji1.classList.add('rock');
  } else if (isPaper(emoji1) && isRock(emoji2)) {
    emoji2.textContent = '?';
    emoji2.classList.remove('rock');
    emoji2.classList.add('paper');
  } else if (isRock(emoji1) && isPaper(emoji2)) {
    emoji1.textContent = '?';
    emoji1.classList.remove('rock');
    emoji1.classList.add('paper');
  }
  // 注意:原文中存在重复条件(如两次 'rock & scissors'),已合并修正
}

? 关键点:classList.remove() 和 classList.add() 是原生、高效且无副作用的操作;避免使用 className = '...' 全量赋值,以防意外覆盖其他动态添加的类(如动画类)。

Tweeze
Tweeze

Tweeze.app是一个AI驱动的个性化新闻简报服务,定位为个人互联网AI阅读助手

下载

✅ 方案二:使用状态变量替代 DOM 查询(高性能,适合大规模 emoji)

若 emoji 数量较多(如数百个),频繁调用 querySelectorAll() 可能带来轻微性能开销。更优做法是用三个状态变量集中维护计数,并在每次状态变更时原子化更新

// 初始化计数器(在 startButton.click 回调内、生成 emoji 后)
let scissorsCount = emojiCount;
let rockCount     = emojiCount;
let paperCount    = emojiCount;

// 在 updateState() 中,每发生一次转换,同步更新变量
if (isScissors(emoji1) && isPaper(emoji2)) {
  emoji2.textContent = '✂️';
  emoji2.classList.remove('paper').add('scissors');
  paperCount--;
  scissorsCount++;
} 
// ... 其他分支同理,统一维护三变量

// updateCounters() 简化为纯状态渲染
function updateCounters() {
  scissorsCountElement.textContent = scissorsCount;
  rockCountElement.textContent     = rockCount;
  paperCountElement.textContent    = paperCount;
}

此方式将「状态」与「视图」分离,符合现代前端设计原则,且 updateCounters() 执行效率恒定 O(1)。

⚠️ 额外注意事项

  • Emoji 字符一致性:原文中使用 ? 表示 rock/paper,易导致混淆。请统一使用标准 Unicode emoji:?(U+1FAA8)、?(U+1F4C4)、✂️(U+2702 FE0F)。确保 HTML、JS、CSS 全链路一致。
  • 初始类名注入:当前循环中每个 emoji 同时添加了 .scissors、.rock、.paper 三类(innerHTML += '<div class="emoji scissors">✂️</div>' + ...),这会导致初始计数错误。应改为按顺序分配单一类型:
    for (let i = 0; i < emojiCount; i++) {
      emojisContainer.innerHTML += `<div class="emoji scissors">✂️</div>`;
      emojisContainer.innerHTML += `<div class="emoji rock">?</div>`;
      emojisContainer.innerHTML += `<div class="emoji paper">?</div>`;
    }
  • 避免重复 setInterval:代码末尾存在 internal = setInterval(...)(疑似拼写错误,应为 interval),且与 playButton 内的 setInterval 重复启动。请删除冗余定时器,仅保留 playButton 中的逻辑。

✅ 总结

updateCounters() 失效的根本原因不是函数本身有误,而是 DOM 结构(class)与业务状态(emoji 类型)未保持同步。修复核心在于:每一次 emoji 类型变更,都必须双向同步——既更新 textContent,也更新 className。方案一更直观易维护;方案二更高效可扩展。二者均能彻底解决计数器“冻结”问题,让屏幕上的战况实时、准确地呈现在顶部计数栏中。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

931

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

32

2025.12.06

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

931

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

32

2025.12.06

js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

531

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

576

2023.07.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

js是什么意思
js是什么意思

JS是JavaScript的缩写,它是一种广泛应用于网页开发的脚本语言。JavaScript是一种解释性的、基于对象和事件驱动的编程语言,通常用于为网页增加交互性和动态性。它可以在网页上实现复杂的功能和效果,如表单验证、页面元素操作、动画效果、数据交互等。

6310

2023.08.17

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

69

2026.03.13

热门下载

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

精品课程

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

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