国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

React中的setState使用細(xì)節(jié)和原理解析

這篇具有很好參考價(jià)值的文章主要介紹了React中的setState使用細(xì)節(jié)和原理解析。希望對(duì)大家有所幫助。如果存在錯(cuò)誤或未考慮完全的地方,請(qǐng)大家不吝賜教,您也可以點(diǎn)擊"舉報(bào)違法"按鈕提交疑問(wèn)。

setState使用詳解

前面我們有使用過(guò)setState的基本使用, 接下來(lái)我們對(duì)setState使用進(jìn)行詳細(xì)的介紹

使用setState的原因

開(kāi)發(fā)中我們并不能直接通過(guò)修改state的值來(lái)讓界面發(fā)生更新

因?yàn)槲覀冃薷牧藄tate之后,希望React根據(jù)最新的State來(lái)重新渲染界面,但是this.state這種方式的修改React并不知道數(shù)據(jù)發(fā)生了變化;

React并沒(méi)有實(shí)現(xiàn)類似于Vue2中的Object.defineProperty或者Vue3中的Proxy的方式來(lái)監(jiān)聽(tīng)數(shù)據(jù)的變化(也就是說(shuō)React并沒(méi)有類似于Vue的數(shù)據(jù)劫持的);

我們必須通過(guò)setState來(lái)告知React數(shù)據(jù)已經(jīng)發(fā)生了變化;

大家可能會(huì)有疑惑:在組件中并沒(méi)有實(shí)現(xiàn)setState的方法,為什么可以調(diào)用呢

原因很簡(jiǎn)單,setState方法是從Component中繼承過(guò)來(lái)的。

我有找到對(duì)應(yīng)的源碼, 可以看到源碼中, setState是放在Component的原型上的

react setstate,React,react.js,javascript,前端

setState的基本用法

setState常見(jiàn)的用法

用法一: 直接在setState函數(shù)中傳入一個(gè)對(duì)象, 傳入的該對(duì)象會(huì)和this.state的對(duì)象進(jìn)行一個(gè)合并, 相同的屬性會(huì)進(jìn)行覆蓋, 修改完成后等待任務(wù)隊(duì)列批處理調(diào)用render函數(shù)實(shí)現(xiàn)頁(yè)面更新

export class App extends Component {
  constructor() {
    super()

    this.state = {
      message: "Hello World"
    }
  }
  
  changeText() {
    this.setState({
      message: "你好啊"
    })
  }

  render() {
    const { message } = this.state
    return (
      <div>
        <h2>{message}</h2>
        <button onClick={() => {this.changeText()}}>按鈕</button>
      </div>
    )
  }
}

用法二: 在setState函數(shù)中傳入一個(gè)回調(diào)函數(shù), 要求傳入的回調(diào)函數(shù)返回一個(gè)對(duì)象, React內(nèi)部會(huì)將返回的這個(gè)對(duì)象和state對(duì)象進(jìn)行合并

這樣的做法有兩個(gè)好處:

好處一: 可以在該回調(diào)函數(shù)中, 編寫修改state數(shù)據(jù)的邏輯, 具有更強(qiáng)的內(nèi)聚性

好處二: 當(dāng)前回調(diào)函數(shù)會(huì)默認(rèn)將state和props傳入進(jìn)來(lái), 我們可以在回調(diào)函數(shù)中直接獲取, 不需要再通過(guò)this.statethis.props的方式獲取;

export class App extends Component {
  constructor() {
    super()

    this.state = {
      message: "Hello World"
    }
  }
  
  changeText() {
    this.setState((state, props) => {
      // 可以直接在回調(diào)函數(shù)中獲取state和props
      console.log(state, props);

      // 對(duì)數(shù)據(jù)進(jìn)行操作的邏輯
      const message = state.message + "Hello React"

      // 該回調(diào)函數(shù)返回一個(gè)對(duì)象
      return {
        message
      }
    })
  }

  render() {
    const { message } = this.state
    return (
      <div>
        <h2>{message}</h2>
        <button onClick={() => {this.changeText()}}>按鈕</button>
      </div>
    )
  }
}

setState的異步更新

setState的更新是異步的嗎?

我們來(lái)驗(yàn)證一下是否真的是異步的, 如下代碼, 在執(zhí)行完setState后, 我們立即打印一下message的結(jié)果

export class App extends Component {
  constructor() {
    super()

    this.state = {
      message: "Hello World"
    }
  }
  
  changeText() {
    // 用法一
    this.setState({
      message: "你好啊"
    })
    console.log(this.state.message) // Hello World
  }

  render() {
    const { message } = this.state
    return (
      <div>
        <h2>{message}</h2>
        <button onClick={() => {this.changeText()}}>按鈕</button>
      </div>
    )
  }
}

打印結(jié)果為Hello World, 并不是修改后的結(jié)果, 可見(jiàn)setState是異步的操作,我們并不能在執(zhí)行完setState之后立馬拿到最新的state的結(jié)果

那么為什么setState設(shè)計(jì)為異步呢?

setState設(shè)計(jì)為異步其實(shí)之前在GitHub上也有很多的討論, 也有許多人會(huì)有同樣的疑惑;

React核心成員(Redux的作者)Dan Abramov也有對(duì)應(yīng)的回復(fù),有興趣的同學(xué)可以參考一下;

https://github.com/facebook/react/issues/11527#issuecomment-360199710;

個(gè)人總結(jié)在Dan Abramov的回答中有說(shuō)到設(shè)計(jì)為異步的兩個(gè)主要原因

原因一: setState設(shè)計(jì)為異步,可以顯著的提升性能;

我們知道調(diào)用setState會(huì)讓render函數(shù)重新執(zhí)行, 如果每次調(diào)用 setState都進(jìn)行一次更新,那么意味著render函數(shù)會(huì)被頻繁調(diào)用,界面重新渲染,這樣效率是很低的;

最好的辦法應(yīng)該是獲取到多個(gè)更新,之后進(jìn)行批量更新(或者說(shuō)統(tǒng)一的更新);

而React中的做法也是如此, 在一個(gè)時(shí)間段中, 會(huì)獲取多個(gè)更新, 再將多個(gè)更新放入一個(gè)任務(wù)隊(duì)列中, 再對(duì)任務(wù)隊(duì)列進(jìn)行批處理; 如果還有其他更新不在當(dāng)前時(shí)間段, 則在下一個(gè)時(shí)間段(或者其他時(shí)間段)的任務(wù)隊(duì)列中進(jìn)行批處理

  • 例如下面代碼, 當(dāng)點(diǎn)擊按鈕時(shí), render函數(shù)只會(huì)執(zhí)行一次, 由此可見(jiàn)是有等待多個(gè)更新再進(jìn)行批處理的
export class App extends Component {
  constructor() {
    super()

    this.state = {
      message: "Hello World"
    }
  }
  
  changeText() {
    this.setState({
      message: "你"
    })
    this.setState({
      message: "你好"
    })
    this.setState({
      message: "你好啊"
    })
  }

  render() {
    console.log("render函數(shù)執(zhí)行")
    const { message } = this.state
    return (
      <div>
        <h2>{message}</h2>
        <button onClick={() => {this.changeText()}}>按鈕</button>
      </div>
    )
  }
}

原因二: 如果同步更新了state,但是還沒(méi)有執(zhí)行render函數(shù),那么state和props不能保持同步;

state和props不能保持一致性,會(huì)在開(kāi)發(fā)中產(chǎn)生很多的問(wèn)題;

例如一個(gè)數(shù)據(jù)message有被展示到頁(yè)面中, 并且傳入到子組件中展示, 此時(shí)通過(guò)setState修改message, 如果是同步的修改完成后, message的值就被改變了; 并且修改完成的后續(xù)代碼有報(bào)錯(cuò)的情況, 在這個(gè)時(shí)候再進(jìn)行調(diào)試的時(shí)候, 會(huì)出現(xiàn)頁(yè)面中message的值被修改掉, 而傳入到子組件的message并沒(méi)有修改的情況, 導(dǎo)致于state和props中的數(shù)據(jù)不一致


setState獲取異步結(jié)果

那么如何可以獲取到更新后的值呢?

方式一:setState的回調(diào)

setState接受兩個(gè)參數(shù):第一個(gè)參數(shù)我們剛剛講解過(guò)了, 第二個(gè)參數(shù)是一個(gè)回調(diào)函數(shù),這個(gè)回調(diào)函數(shù)會(huì)在setState異步有了更新更新后會(huì)執(zhí)行;

格式如下:setState(partialState, callback)

export class App extends Component {
  constructor() {
    super()

    this.state = {
      message: "Hello World"
    }
  }
  
  changeText() {
    // 參數(shù)二回調(diào)函數(shù)可以保證拿到的數(shù)據(jù)是更新后的
    this.setState({ message: "你好啊" }, () => {
      console.log(this.state.message) // 你好啊
    })
  }

  render() {
    console.log("render函數(shù)執(zhí)行")
    const { message } = this.state
    return (
      <div>
        <h2>{message}</h2>
        <button onClick={() => {this.changeText()}}>按鈕</button>
      </div>
    )
  }
}

當(dāng)然,我們也可以在生命周期函數(shù)

componentDidUpdate() {
  console.log(this.state.message);
}

setState一定是異步?

setState一定是異步操作嗎? 其實(shí)在React 18之前分成兩種情況

情況一: 在組件生命周期或React的事件中,setState是異步;

情況二: 在setTimeout或者原生dom事件中,setState是同步;

changeText() {
  	// setTimeout
    setTimeout(() => {
      this.setState({ message: "你好啊" })
      console.log(message) // 你好啊
    }, 0);
  }
  • 在原生dom事件回調(diào)和promise回調(diào), 都是同步的, 這里不再演示****

在React18之后,默認(rèn)所有的操作都被放到了批處理中(也就是默認(rèn)所有操作都是異步處理的), 在官方文檔中有說(shuō)明

react setstate,React,react.js,javascript,前端

在React 18之后, 如果希望代碼可以同步會(huì)拿到,則需要執(zhí)行特殊的flushSync操作:

注意: 使用flushSync需要在react-dom包里面導(dǎo)入

import { flushSync } from 'react-dom';

flushSync要求傳入一個(gè)回調(diào)函數(shù), 把要更新的數(shù)據(jù)放在回調(diào)函數(shù)中, 這樣就是同步的文章來(lái)源地址http://www.zghlxwxcb.cn/news/detail-779642.html

flushSync(() => {
  this.setState({ message: "你好啊" })
})
// 這里獲取就是同步的
console.log(this.state.message) // 你好啊

到了這里,關(guān)于React中的setState使用細(xì)節(jié)和原理解析的文章就介紹完了。如果您還想了解更多內(nèi)容,請(qǐng)?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來(lái)自互聯(lián)網(wǎng)用戶投稿,該文觀點(diǎn)僅代表作者本人,不代表本站立場(chǎng)。本站僅提供信息存儲(chǔ)空間服務(wù),不擁有所有權(quán),不承擔(dān)相關(guān)法律責(zé)任。如若轉(zhuǎn)載,請(qǐng)注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實(shí)不符,請(qǐng)點(diǎn)擊違法舉報(bào)進(jìn)行投訴反饋,一經(jīng)查實(shí),立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費(fèi)用

相關(guān)文章

  • react中使用shouldComponentUpdate生命周期函數(shù)調(diào)用setState引起的無(wú)限循環(huán)的錯(cuò)誤

    在 React 組件中,當(dāng)在 componentWillUpdate 或 componentDidUpdate 生命周期方法中調(diào)用 setState 時(shí),會(huì)觸發(fā)無(wú)限循環(huán),導(dǎo)致超過(guò)最大更新深度。 在React組件中 ,我們使用componentWillUpdate 或 componentDidUpdate 生命周期方法中調(diào)用 setState 方法,但是調(diào)用seState會(huì)導(dǎo)致組件重新渲染 ,那么這個(gè)生命

    2024年02月12日
    瀏覽(23)
  • 【React】組件生命周期、組件通信、setState

    【React】組件生命周期、組件通信、setState

    ? 組件化思想的應(yīng)用: ? ? 有了組件化的思想,我們?cè)谥蟮拈_(kāi)發(fā)中就要充分的利用它。 ? ? 盡可能的將頁(yè)面拆分成一個(gè)個(gè)小的、可復(fù)用的組件。 ? ? 這樣讓我們的代碼更加方便組織和管理,并且擴(kuò)展性也更強(qiáng)。 ? React的組件相對(duì)于Vue更加的靈活和多樣,按照不同的

    2024年01月20日
    瀏覽(20)
  • React中setState是同步還是異步的

    setState()同步、異步總結(jié) 異步的情況: 由React控制的事件處理函數(shù),以及生命周期函數(shù)調(diào)用setState時(shí)表現(xiàn)為異步 。 大部分開(kāi)發(fā)中用到的都是React封裝的事件,比如onChange、onClick、onTouchMove等(合成事件中),這些事件處理函數(shù)中的setState都是異步處理的。 注:上面的事件都是

    2024年02月11日
    瀏覽(17)
  • React 之 內(nèi)置方法setState改變state(一)

    this.setState 方法是React組件類(React.Component 的子類)的一個(gè)內(nèi)置方法。當(dāng)你在創(chuàng)建一個(gè)React組件類時(shí),你繼承自 React.Component,因此你的組件類會(huì)自動(dòng)獲得this.setState 方法。 this.setState 用于更新組件的state。當(dāng)state更新時(shí),React會(huì)重新渲染該組件及其子組件。

    2024年04月28日
    瀏覽(20)
  • react如何處理setState,useState異步問(wèn)題

    flushSync 是 React 提供的一種實(shí)驗(yàn)性的 API,用于控制 React 更新的同步/異步方式,并且只能在 React 16 及更高版本中使用。使用 flushSync 可以強(qiáng)制 React 在執(zhí)行一些特定的 DOM 操作時(shí),同步地(而非異步地)執(zhí)行分塊更新,從而保證消息優(yōu)先級(jí)和交互響應(yīng)性能。 通常情況下,React 采

    2024年02月11日
    瀏覽(17)
  • react中useState、setState、usemeno、meno區(qū)別

    useState和setState是異步 useState : useState 是React函數(shù)組件中的鉤子,用于聲明狀態(tài)變量。 通過(guò) useState ,你可以在函數(shù)組件中添加狀態(tài),而無(wú)需創(chuàng)建類組件。 useState 返回一個(gè)數(shù)組,其中包含當(dāng)前狀態(tài)和一個(gè)更新?tīng)顟B(tài)的函數(shù) setState : setState 是類組件中用于更新?tīng)顟B(tài)的方法。 在類

    2024年02月22日
    瀏覽(22)
  • React - 請(qǐng)你說(shuō)一說(shuō)setState是同步的還是異步的

    React - 請(qǐng)你說(shuō)一說(shuō)setState是同步的還是異步的

    難度級(jí)別:中高級(jí)及以上 ??????????????????????????????提問(wèn)概率:70%? 在React項(xiàng)目中,使用setState可以更新?tīng)顟B(tài)數(shù)據(jù),而不能直接使用為this.state賦值的方式。而為了避免重復(fù)更新state數(shù)據(jù),React首先將state添加到狀態(tài)隊(duì)列中,此時(shí)我們可以通過(guò)shouldComponentUpdate這

    2024年04月09日
    瀏覽(20)
  • 面試題更新之-hook中setState原理

    面試題更新之-hook中setState原理

    在React中,Hook是一種用于在函數(shù)組件中添加狀態(tài)和其他React特性的函數(shù)。它們被引入到React 16.8版本中,旨在解決使用類組件編寫復(fù)雜邏輯時(shí)出現(xiàn)的一些問(wèn)題。 使用Hook,你可以在無(wú)需編寫類的情況下,在函數(shù)組件中使用狀態(tài)(State)、生命周期方法、上下文(Context)等React特性。最

    2024年02月16日
    瀏覽(15)
  • React入門 - 07(說(shuō)一說(shuō) JSX 中的語(yǔ)法細(xì)節(jié))

    React入門 - 07(說(shuō)一說(shuō) JSX 中的語(yǔ)法細(xì)節(jié))

    本章內(nèi)容 上一節(jié)內(nèi)容我們完成了一個(gè)簡(jiǎn)單的 TodoList 案例。到現(xiàn)在為止我們已經(jīng)知道怎么在 JSX 中使用 “js 表達(dá)式”和”列表渲染“了,本節(jié)我們繼續(xù)使用之前的工程項(xiàng)目來(lái)編寫代碼,補(bǔ)充在 React 使用 JSX 語(yǔ)法的一些細(xì)節(jié)。 JSX 中使用表達(dá)式: { JS 表達(dá)式 } 可使用的 js 表達(dá)式

    2024年01月17日
    瀏覽(37)
  • React - Redux Hooks的使用細(xì)節(jié)詳解

    Redux中Hooks介紹 在之前的redux開(kāi)發(fā)中,為了讓組件和redux結(jié)合起來(lái),我們使用了react-redux庫(kù)中的connect : 但是這種方式必須使用高階函數(shù)結(jié)合返回的高階組件; 并且必須編寫:mapStateToProps和 mapDispatchToProps映射的函數(shù), 具體使用方式在前面文章有講解; 在Redux7.1開(kāi)始,提供了Hook的方式

    2024年02月02日
    瀏覽(15)

覺(jué)得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請(qǐng)作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包