首页 > 其他分享 >React入门实例:井字棋

React入门实例:井字棋

时间:2024-10-22 10:44:31浏览次数:1  
标签:function 井字棋 const 入门 handleClick React return squares xIsNext

下面是一个官方教程的实例,实现井字棋且可以回到任意一步。这个实例可以接触到 React 概念,包括元素、组件、props 和 state。详细每一步中文官方地址:https://zh-hans.react.dev/learn/tutorial-tic-tac-toe

代码

import { useState } from 'react';
import './App.css';
// Square组件:单个块可点击填充,接收传过来的2个props
function Square({ value, onSquareClick}) {
  return (
    //点击按钮调用父组件的函数
    <button
    className='square'
    onClick={onSquareClick}
    >{value}</button>
  )
}
// Board组件,共9个块,接收传过来的3个props
function Board({xIsNext, squares, onPlay }) {
  //点击块的函数
  function handleClick(i) {
    //如果已经被占了或者已经有胜者就返回
    if (squares[i] || calculateWinner(squares)) {
      return;
    }
    const nextSquares = squares.slice();//创建 squares 数组的副本
    if (xIsNext) {//如果xIsNext是true就往对应块位置填X
      nextSquares[i] = "X";
    } else {//否则填O
      nextSquares[i] = "O";
    }
    onPlay(nextSquares);//更新状态
  }
  function calculateWinner(squares) {//接收的参数是整个squares数组
    const lines = [//成功的情况
      [0, 1, 2],
      [3, 4, 5],
      [6, 7, 8],
      [0, 3, 6],
      [1, 4, 7],
      [2, 5, 8],
      [0, 4, 8],
      [2, 4, 6]
    ];
    for (let i = 0; i < lines.length; i++) {//遍历每一种成功的情况
      const [a, b, c] = lines[i];// 解构成功的三个索引
      //如果9个块中存在上面成功的情况就返回对应块的值
      if (squares[a] && squares[a] === squares[b] && squares[a] === squares[c]) {
        return squares[a];//返回的是X或O
      }
    }
    return null;// 如果没有胜者,返回null
  }
  const winner = calculateWinner(squares)// 计算当前局势的胜者
  let status;
  if (winner) {//如果有成功值就返回成功者
    status = "Winner: " + winner;
  } else {//否则返回下一局是谁
    status = "Next player: " + (xIsNext ? "X" : "O");
  }
  return(
  <>
  <div className="status">{status}</div>
  {/* 创建3行,每行3个Square组件 */}
  <div className="board-row">
        <Square value={squares[0]} onSquareClick={()=>handleClick(0)} />
        <Square value={squares[1]} onSquareClick={()=>handleClick(1)}/>
        <Square value={squares[2]} onSquareClick={()=>handleClick(2)}/>
      </div>
      <div className="board-row">
        <Square value={squares[3]} onSquareClick={()=>handleClick(3)}/>
        <Square value={squares[4]} onSquareClick={()=>handleClick(4)}/>
        <Square value={squares[5]} onSquareClick={()=>handleClick(5)}/>
      </div>
      <div className="board-row">
        <Square value={squares[6]} onSquareClick={()=>handleClick(6)}/>
        <Square value={squares[7]} onSquareClick={()=>handleClick(7)}/>
        <Square value={squares[8]} onSquareClick={()=>handleClick(8)}/>
      </div>
    </>
  )
}
function App(){
  //xIsNext初始化为true,表示X先行
  const [xIsNext, setXIsNext] = useState(true);
  // 初始化9个块的值都设置null
  const [history, setHistory] = useState([Array(9).fill(null)]);
  //用户当前正在查看的步骤
  const [currentMove, setCurrentMove] = useState(0);
  //当前落子的方块是history选中的位置
  const currentSquares = history[currentMove];

  // 处理落子并更新历史记录
  function handlePlay(nextSquares) {
    //回到过去后显示的历史记录是当前移动到的位置加以前的老历史
    const nextHistory = [...history.slice(0, currentMove + 1), nextSquares];
    setHistory(nextHistory);//更新历史记录,在原来的history数组后加上nextHistory
    setCurrentMove(nextHistory.length - 1);//每次落子时,都需要更新 currentMove 以指向最新的历史条目
    setXIsNext(!xIsNext);//更新xIsNext,切换下一个玩家
  }
  // 跳转到指定历史记录
  function jumpTo(nextMove) {
    setCurrentMove(nextMove); // 更新当前移动
    setXIsNext(nextMove % 2 === 0); // 根据移动的奇偶性更新下一个玩家
  }
//遍历历史记录,生成每一步的描述。squares表示每个元素,move表示每个索引
  const moves = history.map((squares, move) => {
    let description;
    if (move > 0) {
      description = 'Go to move #' + move;// 描述下一步
    } else {
      description = 'Go to game start';// 描述回到开始
    }
    return (
      //对于历史的每一步都创建一个li包含一个按钮
      <li key={move}>
        <button onClick={() => jumpTo(move)}>{description}</button>
      </li>
    );
  });
  return(
    <div className="game">
      <div className="game-board">
        <Board xIsNext={xIsNext} squares={currentSquares} onPlay={handlePlay}/>
      </div>
      <div className="game-info">
        <ol>{moves}</ol>{/* 显示历史记录 */}
      </div>
    </div>
  )
 }
export default App;

结果

标签:function,井字棋,const,入门,handleClick,React,return,squares,xIsNext
From: https://www.cnblogs.com/lushuang55/p/18492107

相关文章

  • 高可用之限流 09-guava RateLimiter 入门使用简介 & 源码分析
    限流系列开源组件rate-limit:限流高可用之限流-01-入门介绍高可用之限流-02-如何设计限流框架高可用之限流-03-Semaphore信号量做限流高可用之限流-04-fixedwindow固定窗口高可用之限流-05-slidewindow滑动窗口高可用之限流-06-slidewindow滑动窗口sentinel源码......
  • Rust小练习,编写井字棋
    画叉画圈的游戏通常指的是井字棋(Tic-Tac-Toe),是一个简单的两人游戏,规则如下:游戏规则棋盘:游戏在一个3x3的方格上进行。玩家:有两个玩家,一个用“X”表示,另一个用“O”表示。目标:玩家轮流在空格中填入自己的标记,目标是先在横向、纵向或斜向上连续放置三个相同的标记。胜利条......
  • CSS入门
    CSS层叠样式表样式表优点缺点使用情况控制范围行内样式表书写方便,权重高没有实现样式和结构相分离较少控制一个标签(少)内部样式表部分结构和样式相分离没有彻底分离较多控制一个页面(中)外部样式表完全实现结构和样式相分离需要引入最多控制整个站点......
  • koa2 入门(1)koa-generator 脚手架和 mongoose 使用
    koa2入门(1)koa-generator脚手架和mongoose使用 项目地址:https://github.com/caochangkui/demo/tree/koa2-learn1构建项目1.1安装koa-generator$npminstall-gkoa-generator1.2使用koa-generator生成koa2项目$koa2-eproject(项目名称)(-e代表使用模板引......
  • springcloud入门-项目搭建
    本文主要供小白使用,详述springcloud项目在实战环境中如何搭建以及常见问题的解决方法,各微服务组件的具体使用及原理,后续我会逐步补充。本文后续论述均以如下环境为前提:jdk:1.8spring-boot:2.6.0spring-cloud:2021.0.9一、公共组件搭建1.parent项目搭建1)pom.xml文......
  • Altium Designer 入门基础教程(一)
     有将近一个半月没有更新过博客了,最近在整理AltiumDesigner 入门基础教程,希望喜欢本系列教程的小伙伴可以点点关注和订阅!下面我们开始进入AltiumDesigner课程的介绍。一、AltiumDesigner快捷键1.原理图快捷键: 2.PCB快捷键:  3.自定义快捷键(原理图、......
  • ctfshow-web入门-信息搜集(14)
    1.根据提示:有时候源码里面就能不经意间泄露重要(editor)的信息,默认配置害死人2.我们直接在url后面添加/editor,在flash上传空间里面找到文件空间,爆出了一堆目录3.最终我们在var/www/html/nothinghere/fl000g.txt这个路径找的到了flag,我们在url后面添加nothinghere/fl000g.......
  • ctfshow-web入门-信息搜集(16)
    1.根据题目提示:考察PHP探针php探针是用来探测空间、服务器运行状况和PHP信息用的,探针可以实时查看服务器硬盘资源、内存占用、网卡流量、系统负载、服务器时间等信息。url后缀名添加/tz.php版本是雅黑PHP探针,然后查看phpinfo搜索flag2.在url后面添加上/tz.php3.点击PHPIN......
  • Java消息队列入门详解
    什么是消息队列?消息队列的产生主要是为了解决系统间的异步解耦与确保最终一致性。在实际应用场景中,往往存在一些主流程操作和辅助流程操作,其中主流程需要快速响应用户请求,而辅助流程可能涉及复杂的处理逻辑或者依赖于外部服务。通过将这些辅助流程的消息放入消息队列,使得它们可......
  • P6189 [NOI Online #1 入门组] 跑步(分拆数)
    简要题意给你一个整数\(n\),你需要求\(\sum_{i=1}^nx_i=n\)且\(x_i\lex_{i+1}\)的非负整数解数量对给定模数\(p\)取模后的结果。\(n\le10^5\)分析考虑一个显然的DP:设\(f_{i,j}\)表示考虑\(1\simi\)这些数,总和为\(j\)的方案数。转移是完全背包型转移:\(f_{i,j}......