0

0

HTML如何制作井字棋?胜负判断逻辑怎么写?

幻夢星雲

幻夢星雲

发布时间:2025-08-07 16:00:03

|

947人浏览过

|

来源于php中文网

原创

构建井字棋的html和css基础布局是:1. 使用一个包含9个带data-index属性的div.cell的div#game-board容器来搭建棋盘结构;2. 利用css grid设置3x3网格,每个格子100px,通过flex布局居中内容,并添加边框、悬停效果和x/o不同颜色样式;3. 添加game-status显示区域和reset-button重置按钮,整体布局居中且具备良好视觉反馈,完整实现了功能清晰、结构语义化、样式美观的前端界面。

HTML如何制作井字棋?胜负判断逻辑怎么写?

制作井字棋(Tic-Tac-Toe)的核心在于用HTML搭建棋盘结构,用CSS进行样式美化,而最关键的胜负判断和游戏逻辑,则完全依赖JavaScript来实现。说白了,就是HTML提供骨架,CSS赋予外貌,JavaScript注入灵魂,尤其是那个判断谁赢了的逻辑,是整个游戏的心脏。

解决方案

要构建一个井字棋,我们首先需要一个能代表9个格子的HTML结构,通常是一个简单的

div
容器里面嵌套9个小
div
作为棋盘格。CSS负责把这些
div
排列成3x3的网格,并给它们加上边框,让它们看起来像棋盘。

而真正的魔法发生在JavaScript里。我们需要一个数组来表示棋盘的当前状态,比如

board = ['', '', '', '', '', '', '', '', '']
,每个元素对应一个格子,初始为空。当玩家点击一个格子时,我们更新这个数组对应位置的值('X'或'O'),然后更新HTML页面上那个格子的显示。

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

核心挑战在于,每次玩家落子后,我们都需要检查当前棋盘状态,看看有没有人赢了,或者是不是平局了。这个检查逻辑会遍历所有可能的获胜组合(三行、三列、两条对角线),看是否有玩家连续占据了其中一条线。如果检查到有玩家获胜,就宣告胜利;如果棋盘满了还没有胜利者,那就是平局。

构建井字棋的HTML和CSS基础布局是怎样的?

我个人觉得,一个清晰、语义化的HTML结构是所有前端项目的基础,井字棋也不例外。最直观的方式就是用一个主容器包裹9个代表棋格的元素。

轮到 X 落子

这里,

data-index
属性非常有用,它能让我们在JavaScript中轻松地知道用户点击了哪个格子。

至于CSS,它就像是给这个骨架穿上衣服,让它看起来像个真正的棋盘。使用CSS Grid或Flexbox都能很方便地实现3x3的布局。

#game-board {
    display: grid;
    grid-template-columns: repeat(3, 100px); /* 3列,每列100px宽 */
    grid-template-rows: repeat(3, 100px);    /* 3行,每行100px高 */
    width: 300px;
    height: 300px;
    border: 2px solid #333;
    margin: 20px auto;
    box-shadow: 0 0 10px rgba(0,0,0,0.1);
}

.cell {
    width: 100px;
    height: 100px;
    border: 1px solid #ccc;
    display: flex;
    justify-content: center;
    align-items: center;
    font-size: 4em; /* X或O的字体大小 */
    cursor: pointer;
    transition: background-color 0.3s ease;
}

.cell:hover {
    background-color: #f0f0f0;
}

/* 标记 X 和 O 的样式 */
.cell.x { color: #e74c3c; }
.cell.o { color: #3498db; }

#game-status {
    text-align: center;
    margin-top: 10px;
    font-size: 1.5em;
    font-weight: bold;
}

#reset-button {
    display: block;
    margin: 20px auto;
    padding: 10px 20px;
    font-size: 1em;
    cursor: pointer;
    background-color: #2ecc71;
    color: white;
    border: none;
    border-radius: 5px;
    transition: background-color 0.3s ease;
}

#reset-button:hover {
    background-color: #27ae60;
}

这样的布局,简洁明了,既保证了功能性,也兼顾了视觉上的可读性。

JavaScript如何管理游戏状态和玩家交互?

在我看来,真正让棋盘“活”起来的,是JavaScript对游戏状态的精确管理和对玩家行为的响应。这就像是游戏的“大脑”和“神经系统”。

VWO
VWO

一个A/B测试工具

下载

首先,我们需要几个关键变量来跟踪游戏状态:

let board = ['', '', '', '', '', '', '', '', '']; // 棋盘状态,9个空字符串代表空位
let currentPlayer = 'X'; // 当前玩家,从'X'开始
let gameActive = true; // 游戏是否还在进行中

// 获取DOM元素
const cells = document.querySelectorAll('.cell');
const statusDisplay = document.getElementById('game-status');
const resetButton = document.getElementById('reset-button');

然后,就是处理玩家点击格子的逻辑:

function handleCellClick(clickedCellEvent) {
    const clickedCell = clickedCellEvent.target;
    const clickedCellIndex = parseInt(clickedCell.dataset.index);

    // 如果格子已经被占或者游戏已经结束,就什么也不做
    if (board[clickedCellIndex] !== '' || !gameActive) {
        return;
    }

    // 更新棋盘状态和DOM显示
    board[clickedCellIndex] = currentPlayer;
    clickedCell.innerHTML = currentPlayer;
    clickedCell.classList.add(currentPlayer.toLowerCase()); // 添加类名用于样式

    // 检查胜负或平局
    if (checkWin()) {
        statusDisplay.innerHTML = `玩家 ${currentPlayer} 获胜!`;
        gameActive = false;
        return;
    }

    if (checkDraw()) {
        statusDisplay.innerHTML = `平局!`;
        gameActive = false;
        return;
    }

    // 切换玩家
    currentPlayer = currentPlayer === 'X' ? 'O' : 'X';
    statusDisplay.innerHTML = `轮到 ${currentPlayer} 落子`;
}

// 添加事件监听器
cells.forEach(cell => cell.addEventListener('click', handleCellClick));
resetButton.addEventListener('click', resetGame); // 重置游戏按钮的监听器

resetGame
函数就比较简单了,把所有状态重置回初始值:

function resetGame() {
    board = ['', '', '', '', '', '', '', '', ''];
    currentPlayer = 'X';
    gameActive = true;
    statusDisplay.innerHTML = `轮到 ${currentPlayer} 落子`;
    cells.forEach(cell => {
        cell.innerHTML = '';
        cell.classList.remove('x', 'o'); // 移除X和O的样式
    });
}

通过这些,我们已经构建了一个响应式的游戏交互骨架。

井字棋的胜负判断逻辑如何实现?有哪些常见的判断方法?

在我看来,井字棋的胜负判断,有点像在玩侦探游戏,我们需要列出所有可能的“犯罪现场”(获胜组合),然后逐一排查。这是整个游戏最核心、也最有意思的逻辑部分。

井字棋总共有8种获胜组合:

  1. 三行: (0, 1, 2), (3, 4, 5), (6, 7, 8)
  2. 三列: (0, 3, 6), (1, 4, 7), (2, 5, 8)
  3. 两条对角线: (0, 4, 8), (2, 4, 6)

我们可以把这些组合定义在一个数组中:

const winConditions = [
    [0, 1, 2], // 第一行
    [3, 4, 5], // 第二行
    [6, 7, 8], // 第三行
    [0, 3, 6], // 第一列
    [1, 4, 7], // 第二列
    [2, 5, 8], // 第三列
    [0, 4, 8], // 主对角线
    [2, 4, 6]  // 副对角线
];

有了这个,

checkWin
函数就可以遍历这些组合,检查当前棋盘状态:

function checkWin() {
    for (let i = 0; i < winConditions.length; i++) {
        const condition = winConditions[i];
        // 提取当前组合的三个格子索引
        const a = condition[0];
        const b = condition[1];
        const c = condition[2];

        // 检查这三个格子是否都被同一个玩家占据,并且不为空
        if (board[a] !== '' && board[a] === board[b] && board[b] === board[c]) {
            return true; // 发现胜利者
        }
    }
    return false; // 没有发现胜利者
}

平局的判断相对简单,它发生在所有格子都被填满,但却没有玩家获胜的情况下。

function checkDraw() {
    // 检查棋盘是否所有格子都已填充
    // board.includes('') 会检查数组中是否包含空字符串,如果包含,说明还有空位
    return !board.includes('') && !checkWin();
}

这里,

!board.includes('')
确保了所有格子都已填满。而
!checkWin()
则确保了在填满的情况下,没有出现胜利者。这两者同时满足,才是真正的平局。这种方法既高效又直观,是实现井字棋胜负判断的经典策略。

相关文章

HTML速学教程(入门课程)
HTML速学教程(入门课程)

HTML怎么学习?HTML怎么入门?HTML在哪学?HTML怎么学才快?不用担心,这里为大家提供了HTML速学教程(入门课程),有需要的小伙伴保存下载就能学习啦!

下载

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

相关专题

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

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

557

2023.06.20

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

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

394

2023.07.04

js四舍五入
js四舍五入

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

754

2023.07.04

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

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

478

2023.09.01

JavaScript转义字符
JavaScript转义字符

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

454

2023.09.04

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

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

1031

2023.09.04

如何启用JavaScript
如何启用JavaScript

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

658

2023.09.12

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

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

553

2023.09.20

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

1

2026.01.21

热门下载

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

精品课程

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

共14课时 | 0.8万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.9万人学习

CSS教程
CSS教程

共754课时 | 21.5万人学习

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

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