国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

2023 Vue開發者的React入門

開發 前端
React? 新文檔重新設計了導航結構,讓我們更加輕松地找到所需的文檔和示例代碼 不僅提供了基礎知識的介紹,還提供了更加詳細的原理介紹和最佳實踐,包括:React? 組件的設計哲學、React Hooks的原理和用法等。

Vue 和 React 都是流行的 JavaScript 框架,它們在組件化、數據綁定等方面有很多相似之處

本文默認已有現代前端開發(Vue)背景,關于 組件化、前端路由、狀態管理 概念不會過多介紹

0基礎建議詳細閱讀 Thinking in React-官方文檔 了解 React 的設計哲學

  • React 新文檔- https://react.dev
  • React 中文文檔(翻譯中)- https://react.jscn.org

經過本文的學習讓沒開發過 React 項目的 Vue 開發者可以上手開發現有的 React 項目,完成工作需求開發

React 新文檔

React 新文檔重新設計了導航結構,讓我們更加輕松地找到所需的文檔和示例代碼 不僅提供了基礎知識的介紹,還提供了更加詳細的原理介紹和最佳實踐,包括:React 組件的設計哲學、React Hooks的原理和用法等

并且提供了在線編輯和運行的功能,方便開發者進行測試和實驗

?? 基于 函數組件

圖片

初學可以只學 函數組件,You Don't Need to Learn Class Components

圖片

?? interactive sandboxes 可交互沙箱,邊做邊學

圖片

Fork 可以單獨打開頁簽

圖片

JSX 與 SFC

  • 在 Vue 中我們使用 單文件組件(SFC) 編寫組件模版 (雖然 Vue 也支持使用 JSX , 但是更鼓勵使用SFC)
  • 在 React 中,JSX(JavaScript XML)是一種將HTML語法嵌入到 JavaScript 中的語法擴展。它可以使得我們在 JavaScript 代碼中輕松地定義組件的結構和樣式,從而提高代碼的可讀性和可維護性

雖然 React和 Vue 在組件定義方式上存在差異,但是它們的組件化思想是相似的

根節點

?? Vue

<template>
  <div>同級節點1</div>
  <div>同級節點2</div>
</template>

?? React

const App = (
  <>
    <div>同級節點1</div>
    <div>同級節點2</div>
  </>
)

const App = (
  <React.Fragment>
    <div>同級節點1</div>
    <div>同級節點2</div>
  </React.Fragment>
)

條件渲染

?? Vue

<div v-if="show">條件渲染</div>
<div v-show="show">條件渲染</div>

?? React

{
  show ? <div>條件渲染</div> : null
}

循環語句

?? Vue

<ul>
  <li v-for="i in list" :key="i.id">{i.name}</li>
</ul>

?? React

<ul>
  { list.map(i => <li key={i.id}>{i.name}</li>) }
</ul>

表單綁定

?? Vue

<input v-model="value"/>

?? React

<input value={value} notallow={onChange}/>

可以看出 React 的 JSX語法 學習記憶成本更低一點(當然Vue也不復雜),Vue 更語法糖一些

單向數據流與雙向綁定

在 Vue 中,我們使用 v-bind、v-modal對數據進行綁定,無論是來自用戶操作導致的變更,還是在某個方法里賦值都能夠直接更新數據,不需要手動進行 update 操作

this.data.msg = '直接修改數據后視圖更新'

在 React 中,數據流是單向的,即從父組件傳遞到子組件,而不允許子組件直接修改父組件的數據。需要調用set 方法更新,當 React 感應到 set 觸發時會再次調用 render 對 dom 進行刷新

msg = "Hello" // ? 錯誤寫法

setMsg('Hello'); // ? 來自hooks的set寫法 后面會介紹

?? Vue 本質上底層也是單向的數據流,只不過對使用者來說看起來是雙向的,如 v-model 本質也要 set

React Hooks

React Hooks 是 React 16.8 版本中引入的特性,它可以讓我們在 函數組件 中使用狀態(state)和其他 React 特性

Hooks 本質是一些管理組件狀態和邏輯的 API ,它允許開發者在 函數式組件 中使用狀態、副作用和鉤子函數,可以更加方便地管理組件狀態、響應式地更新DOM、使用上下文等

在沒有 Hooks 前, 函數組件 不能擁有狀態,只能做簡單功能的UI(靜態元素展示),大家使用 類組件 來做狀態組件

因為 函數組件 更加匹配 React 的設計理念 UI = f(data),也更有利于邏輯拆分與重用的組件表達形式,為了能讓 函數組件 可以擁有自己的狀態,Hooks 應運而生

組件的邏輯復用

在 Hooks 出現之前,React 先后嘗試了 mixins混入,HOC高階組件,render-props等模式。但是都有各自的問題,比如 mixins 的數據來源不清晰,高階組件的嵌套問題等等

class組件自身的問題

class組件就像一個厚重的‘戰艦’ 一樣,大而全,提供了很多東西,有不可忽視的學習成本,比如各種生命周期,this指向問題等等

useState

參數接受一個默認值,返回 [value, setValue] 的元組(就是約定好值的 JavaScript 數組),來讀取和修改數據

?? 不使用 Hooks 的靜態組件,當點擊修改數據,視圖不會重新渲染

function App() {
  let count = 1
  const add = () => count++ // 不會觸發重新渲染

  return <div onClick={add}>{count}</div>
}

?? 使用 useState

import { useState } from 'react'

function App() {
  let count = 1
  const [proxyCount, setProxyCount] = useState(count)
  const add = () => setProxyCount(proxyCount+1)

  return <div onClick={add}>{proxyCount}</div>
}

我們分析一下觸發數據修改的 函數組件行為:

組件會第二次渲染(useState 返回的數組第二項 setProxyCount() 被執行就會觸發重新渲染)

  1. 點擊按鈕,調用 setProxyCount(count + 1) 修改狀態,因為狀態發生改變,所以,該組件會重新渲染
  2. 組件重新渲染時,會再次執行該組件中的代碼邏輯
  3. 再次調用 useState(1),此時 React 內部會拿到最新的狀態值而非初始值,比如,該案例中最新的狀態值為 2
  4. 再次渲染組件,此時,獲取到的狀態 count 值為 2

?? 也就是觸發重新渲染會讓 useState 也重新執行,但是 useState 的參數(初始值)只會在組件第一次渲染時生效

每次的渲染,useState 獲取到都是最新的狀態值,React 組件會記住每次最新的狀態值

useEffect

上面我們分析觸發組件重新渲染就可以發現,React 的函數組件沒有具體的生命周期鉤子

React 更希望我們把組件當作函數,而去關注函數的函數的副作用,而沒有實例化過程的鉤子

useEffect 就可以很好的幫助我們達到我們想要的效果:

  1. 處理組件第一次渲染時的回調,類似 Vue 中的 mounted
// 第二個參數傳一個空數組,表示沒有依賴,只會在第一次渲染時執行
useEffect(() => {
  alert('mounted');
}, [])
  1. 通過依賴變更觸發的鉤子函數,只要有一項依賴發生變化就執行,類似 Vue 中的 watch
function Comp({ title }) {
  const [count, setCount] = useState(0);
  // 第二個參數指定一個數組,放入你想監聽的依賴:
  useEffect(() => {
    console.log('title or count has changed.')
  }, [title, count])
}

原則上,函數中用到的所有依賴都應該放進數組里

  1. 組件卸載時執行內部 return 的函數
import { useEffect } from "react"

const App = () => {

  useEffect(() => {
    const timerId = setInterval(() => {
      console.log('定時器在運行')
    }, 1000)

    return () => { // 用來清理副作用的事情
      clearInterval(timerId)
    }
  }, [])

  return <div>內部有定時器</div>
}

我們常見的副作用 1. 數據請求ajax發送 2. 手動修改dom 3. localstorage操作

自定義 Hooks

獲取滾動距離y:

import { useState, useEffect } from "react"

export function useWindowScroll () {
  const [y, setY] = useState(0)

  useEffect(() => {
    const scrollHandler = () => {
      const h = document.documentElement.scrollTop
      setY(h)
    }
    window.addEventListener('scroll', scrollHandler)
    return () => window.removeEventListener('scroll', scrollHandler)
  })

  return [y]
}

使用:

const [y] = useWindowScroll()
return <div>{y}</div>

圖片

封裝的 Hooks 名稱也要用 use 開頭(這是一個約束)

狀態管理

React 的 狀態管理 有很多,入門可以暫時不考慮

或者已有項目使用什么再學習即可,和 Vuex 整體思路差不多

tic-tac-toe 井字棋游戲

最后我們跟著 React 官方文檔實現一個井字棋游戲來鞏固知識點

使用 Vite 創建項目

圖片

pnpm create vite react-tic-tac-toe --template react
cd react-tic-tac-toe
pnpm i
pnpm dev

?? vite.config.js 非常簡潔

import { defineConfig } from 'vite'
import react from '@vitejs/plugin-react'

export default defineConfig({
  plugins: [react()],
})

?? 修改入口文件 main.jsx

import React, { StrictMode } from "react";
import { createRoot } from "react-dom/client";
import "./index.css";

import App from "./App";

const root = createRoot(document.getElementById("root"));
root.render(
  <StrictMode>
    <App />
  </StrictMode>
);

?? util.js 計算當前棋局是否有獲勝

// 計算當前棋局是否有獲勝
export function calculateWinner(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];
    if (squares[a] && squares[a] === squares[b] && squares[a] === squares[c]) {
      return squares[a];
    }
  }
  return null;
}

?? Square.jsx 正方形按鈕組件

// 正方形按鈕組件
export default function Square({ value, onSquareClick }) {
  return (
    <button className="square" onClick={onSquareClick}>
      {value}
    </button>
  );
}

?? App.jsx

import { useState } from 'react';
import { calculateWinner } from './util.js'
import Square from './Square'

function Board({ xIsNext, squares, onPlay }) {
  function handleClick(i) {
    if (calculateWinner(squares) || squares[i]) {
      return;
    }
    const nextSquares = squares.slice();
    if (xIsNext) {
      nextSquares[i] = 'X';
    } else {
      nextSquares[i] = 'O';
    }
    // 執行父組件的落子事件
    onPlay(nextSquares);
  }

  const winner = calculateWinner(squares);
  let status;
  if (winner) {
    // 勝利提示
    status = '獲勝方是: ' + winner;
  } else {
    // 下一步提示
    status = 'Next player: ' + (xIsNext ? 'X' : 'O');
  }

  return (
    <>
      <div className="status">{status}</div>
      <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>
    </>
  );
}

export default function Game() {
  const [history, setHistory] = useState([Array(9).fill(null)]);
  const [currentMove, setCurrentMove] = useState(0);
  const xIsNext = currentMove % 2 === 0;
  const currentSquares = history[currentMove];

  // 棋盤落子
  function handlePlay(nextSquares) {
    const nextHistory = [...history.slice(0, currentMove + 1), nextSquares];
    // 記錄落子歷史,用于恢復棋局
    setHistory(nextHistory);
    setCurrentMove(nextHistory.length - 1);
  }

  // 恢復棋局到第幾步
  function jumpTo(nextMove) {
    setCurrentMove(nextMove);
  }

  // 歷史落子列表按鈕展示,用于點擊恢復棋局
  const moves = history.map((squares, move) => {
    let description;
    if (move > 0) {
      description = 'Go to move #' + move;
    } else {
      description = 'Go to game start';
    }
    return (
      <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>
  );
}

圖片

深入學習任一前端框架都不容易,讓我們一起加油吧!

參考資料

  • React 新文檔- https://react.dev
  • React 中文文檔(翻譯中)- https://react.jscn.org
  • 給 Vue 開發的 React 上手指南- https://juejin.cn/post/6952545904087793678
  • 無縫切換?從Vue到React- https://zhuanlan.zhihu.com/p/609120596
  • How to Learn React in 2023- https://www.freecodecamp.org/news/how-to-learn-react-in-2023


圖片

責任編輯:武曉燕 來源: 大轉轉FE
相關推薦

2024-05-07 08:45:16

OpenAILlamaIndex大語言模型

2022-08-12 08:02:11

Solid.jsReact

2024-07-22 08:03:55

2019-04-09 15:12:43

開發者技能工具

2012-06-13 01:23:30

開發者程序員

2021-12-24 11:24:59

React HackReact JavaScript

2017-11-27 13:09:00

AndroidGradle代碼

2018-07-18 09:12:05

開發者Java工具

2013-08-30 09:41:46

JavaApache CameApache

2021-08-27 12:59:59

React前端命令

2022-02-13 00:24:33

開發VueJavaScrip

2015-08-06 17:15:28

2014-05-08 13:36:07

Android Wea開發者預覽版

2024-05-09 08:50:39

React編譯器工具

2024-10-12 09:38:53

2025-03-17 08:00:00

2023-08-15 10:10:34

點贊
收藏

51CTO技術棧公眾號

日韩理论片网站| 瑟瑟视频在线| 成人毛片老司机大片| 日韩一区二区在线播放| 精产国产伦理一二三区| 久久久久久免费| 4444亚洲人成无码网在线观看| 亚洲黄色三级| 波多野结衣成人在线| 久久影视一区| 成人h猎奇视频网站| 国产探花在线精品| 69国产精品成人在线播放| 91国产一区| 久久精品99久久香蕉国产色戒| 这里有精品可以观看| 日韩高清免费观看| xxxxx性欧美特大| 亚洲一区www| h1515四虎成人| 色爱av美腿丝袜综合粉嫩av | 一个人看的www久久| 交100部在线观看| 亚洲欧美在线免费| 亚洲成人a级片| 午夜精品三级视频福利| 久久综合欧美| 91视频免费在线| 国产欧美亚洲一区| 在线不卡日本| 91美女片黄在线| 成人伦理视频网站| 色婷婷亚洲一区二区三区| 婷婷在线视频| 亚洲精品在线观看网站| 都市激情亚洲一区| 久久久久久久国产| 国产99精品| 国产福利久久| 狠狠色丁香久久婷婷综合_中| 九色在线视频观看| 一区二区三区在线视频免费| jizz在线观看视频| 国产亚洲福利一区| 日韩精品亚洲aⅴ在线影院| 99精品欧美一区二区三区| 日韩视频一区| 拔插拔插海外华人免费| 亚洲柠檬福利资源导航| 麻豆电影在线播放| 久久这里有精品视频| 欧美gayvideo| 在线看无码的免费网站| 国产精品美女久久久久久久久久久| 污视频在线观看网站| 日韩精品一区二区三区蜜臀| 特级西西444| 亚洲欧洲另类国产综合| 免费观看久久久久| 精品中文字幕视频| 亚洲毛片网站| 久久久久久久片| 欧美视频一区在线观看| 日本高清久久一区二区三区| 成人午夜在线观看视频| 一区二区三区视频免费在线观看 | 午夜激情久久| av电影一区二区三区| 亚洲欧美色一区| 18+激情视频在线| 97在线精品视频| 日韩精品亚洲一区| 婷婷色播视频| 精品五月天久久| 97色伦图片97综合影院| 日本福利视频网站| 日韩欧美高清视频| 欧美成人福利| 激情视频在线观看一区二区三区| 国产精品影视天天线| 尤物网站在线| 久久亚洲影音av资源网| 亚洲三级毛片| 国产一级粉嫩xxxx| 亚洲精品视频网上网址在线观看| 日韩欧美四区| 国产亚洲精品久久久久久久| 大伊人狠狠躁夜夜躁av一区| 日本午夜精品久久久久| 久久久水蜜桃| 亚洲大片在线观看| 亚洲日本va午夜在线电影| 天天做天天爱天天高潮| 欧美日韩中文精品| 精品国产一区二区三区小蝌蚪| av动漫在线播放| 欧美美女一区二区三区| 欧美综合视频| 成人天堂av| 久久综合色影院| 国产一区二区三区在线看麻豆| yiren22亚洲综合伊人22| 国产福利精品av综合导导航| 91麻豆免费看| 二吊插入一穴一区二区| 日本一区视频在线播放| 在线免费视频一区二区| 欧美精品久久久久久| 超碰影院在线观看| 丝袜一区二区三区| 国产精品911| 黄频免费在线观看| 欧美日韩国产精品一区二区| 色综合久久中文综合久久97| 国产亚洲一区| 亚洲污视频在线观看| 久久成人一区二区| 国产精品99久久久久久有的能看| www.51av欧美视频| 一区二区三区av在线| 亚洲成av人影院在线观看| 肉肉av福利一精品导航| 操你啦视频在线| 欧美日韩精品中文字幕一区二区| 欧美日韩第一区日日骚| 亚洲精选久久| 亚洲wwwww| 正在播放精油久久| 精品亚洲永久免费精品| 精品一区二区在线看| 人人草在线视频| 黄色污污在线观看| 中文字幕精品视频| 99riav久久精品riav| 免费精品一区二区三区在线观看| 97在线播放视频| 欧美精品videos| 亚洲美女偷拍久久| 希岛爱理av一区二区三区| www.av在线| 杨幂一区欧美专区| 亚洲图片制服诱惑| 91视频观看视频| 任你躁在线精品免费| 香蕉视频色版| 国产精品久久久对白| 欧美日韩三级一区二区| 日韩电影一区二区三区| 午夜伦理福利在线| 欧美大尺度激情区在线播放| 国产高清精品网站| 欧美一级大片在线视频| 日本aⅴ中文| 精品伦精品一区二区三区视频| 精品美女在线播放| 成人精品免费网站| 亚洲精品国偷自产在线99热| 国产盗摄精品一区二区三区在线| 亚洲ww精品| 久草在线在线| 欧美一区免费视频| 色综合伊人色综合网| 亚洲免费电影在线| 一区二区三区高清视频在线观看| 日韩伦理在线一区| 男人的天堂日韩| 国产精品欧美久久久| 日韩欧美一区中文| ww亚洲ww在线观看国产| 欧美午夜精品一区二区三区电影| 欧美人xxx| 精品视频免费在线播放| 奇米一区二区三区四区久久| 在线观看日韩毛片| 国产精品1区2区| 精品日韩欧美一区| 99thz桃花论族在线播放| 日本黄大片一区二区三区| 国产在线精品一区二区三区》| 亚洲视频在线观看视频| 亚洲狠狠爱一区二区三区| 麻豆一区二区三| 国产日产精品一区二区三区四区的观看方式 | 欧美 日韩 国产精品免费观看| 女同视频在线观看| 污网站免费看| 欧美另类一区| 国产综合在线视频| 日韩欧美一级二级三级| 中文字幕av资源一区| 一本久道久久久| 大奶在线精品| 日本中文字幕中出在线| www99xav| 成人在线观看www| 亚洲一区中文字幕| 欧美高跟鞋交xxxxxhd| 亚洲精品在线一区二区| 天天色图综合网| 久久久久综合网|