React類組件和函數(shù)組件對(duì)比-Hooks的簡(jiǎn)介
Hooks的介紹和初體驗(yàn)
Hooks出現(xiàn)的原因
Hook 是 React 16.8 的新增特性,它可以讓我們?cè)诓痪帉慶lass的情況下, 使用state以及其他的React特性(比如生命周期)。
我們先來思考一下class組件相對(duì)于函數(shù)式組件有什么優(yōu)勢(shì)?比較常見的是下面這些優(yōu)勢(shì):
class組件可以定義自己的state,用來保存組件自己內(nèi)部的狀態(tài);
import React, { PureComponent } from 'react' // 類組件 class HellWorld extends PureComponent { constructor() { super() // 類組件可以在state中保存自己的狀態(tài) this.state = { message: "Hello World" } } render() { const { message } = this.state return ( <div> <h2>{message}</h2> </div> ) } } export class App extends PureComponent { render() { return ( <div> <HellWorld/> </div> ) } } export default App
函數(shù)式組件不可以,因?yàn)楹瘮?shù)每次調(diào)用都會(huì)產(chǎn)生新的臨時(shí)變量;
例如下面代碼中, 看起來似乎是有保存狀態(tài), 但是其實(shí)是有兩大致命缺陷的
- 第一: 我們?nèi)绻幸恍┎僮鲃?dòng)態(tài)的修改了message這個(gè)狀態(tài), 其實(shí)這個(gè)函數(shù)組件是不知道要重新渲染的(意味著哪怕修改了message, 頁面也不會(huì)刷新)
- 第二: 就算頁面重現(xiàn)渲染, 意味著這個(gè)函數(shù)組件會(huì)重新執(zhí)行, 那么就會(huì)重新將最初始的值賦值給message(口水話來說就是, 函數(shù)重新執(zhí)行, 修改了也白修改)
import React, { PureComponent } from 'react' // 函數(shù)組件 function HelloHooks() { // 就算函數(shù)重新執(zhí)行, 又會(huì)重新賦值, 無意義 let message = "Hello Hooks" return( <div> <h2>{message}</h2> </div> ) } export class App extends PureComponent { render() { return ( <div> <HellWorld/> <HelloHooks/> </div> ) } } export default App
class組件有自己的生命周期,我們可以在對(duì)應(yīng)的生命周期中完成自己的邏輯;
比如在componentDidMount中發(fā)送網(wǎng)絡(luò)請(qǐng)求,并且該生命周期函數(shù)只會(huì)執(zhí)行一次;
函數(shù)式組件在學(xué)習(xí)hooks之前,是沒有類似于生命周期這樣的函數(shù)
如果在函數(shù)中發(fā)送網(wǎng)絡(luò)請(qǐng)求,意味著每次重新渲染都會(huì)重新發(fā)送一次網(wǎng)絡(luò)請(qǐng)求, 這種不必要的重新發(fā)送網(wǎng)絡(luò)請(qǐng)求是非常影響性能的;
class組件可以在狀態(tài)改變時(shí), 只重新執(zhí)行render函數(shù)以及我們希望重新調(diào)用的生命周期函數(shù);
函數(shù)式組件在重新渲染時(shí),整個(gè)函數(shù)都會(huì)被執(zhí)行,似乎沒有什么地方可以只讓它們調(diào)用一次;
所以,在Hook出現(xiàn)之前,對(duì)于上面這些情況我們通常都會(huì)編寫class組件, 沒辦法在函數(shù)組件中編寫。
類組件存在的問題
復(fù)雜組件變得難以理解:
我們?cè)谧畛蹙帉懸粋€(gè)class組件時(shí),往往邏輯比較簡(jiǎn)單,并不會(huì)非常復(fù)雜。但是隨著業(yè)務(wù)的增多,我們的class組件會(huì)變得越來越復(fù)雜
;
比如componentDidMount中,可能就會(huì)包含大量的邏輯代碼:包括網(wǎng)絡(luò)請(qǐng)求、一些事件的監(jiān)聽(還需要在 componentWillUnmount中移除);
而對(duì)于這樣的class實(shí)際上非常難以拆分:因?yàn)樗鼈兊倪壿嬐煸谝黄穑瑥?qiáng)行拆分反而會(huì)造成過度設(shè)計(jì),增加代碼的復(fù)雜度;
難以理解的class:
對(duì)于有些人來說學(xué)習(xí)ES6的class也 是學(xué)習(xí)React的一個(gè)障礙。
比如在class中,我們必須搞清楚this的指向到底是誰,所以需要花很多的精力去學(xué)習(xí)this;
雖然我認(rèn)為前端開發(fā)人員必須掌握this,但是依然處理起來非常麻煩;
組件復(fù)用狀態(tài)很難:
在前面為了一些狀態(tài)的復(fù)用我們需要通過高階組件;
像我們之前學(xué)習(xí)的redux中connect或者react-router(6版本之前)中的withRouter,這些高階組件設(shè)計(jì)的目的就是為了狀態(tài)的復(fù)用;
或者類似于Provider、Consumer來共享一些狀態(tài),但是多次使用Consumer時(shí),我們的代碼就會(huì)存在很多嵌套;
這些代碼讓我們不管是編寫和設(shè)計(jì)上來說,都變得非常困難;
Hooks的使用注意
Hook的出現(xiàn),可以解決上面提到的這些問題;
簡(jiǎn)單總結(jié)一下hooks:
它可以讓我們?cè)诓痪帉慶lass的情況下, 使用state以及其他的React特性(意味著不學(xué)習(xí)class關(guān)鍵字和this指向依然可以編寫React);
但是我們可以由此延伸出非常多的用法,來讓我們前面所提到的問題得到解決;
Hook的使用場(chǎng)景:
Hook的出現(xiàn)基本可以代替我們之前所有使用class組件的地方;
但是如果是一個(gè)舊的項(xiàng)目,你并不需要直接將所有的代碼重構(gòu)為Hooks,因?yàn)樗耆蛳录嫒?,你可以漸進(jìn)式的來使用它;
Hook只能在函數(shù)組件中使用,不能在類組件或者函數(shù)組件之外的地方使用;
在我們繼續(xù)學(xué)習(xí)Hooks之前,請(qǐng)記住以下幾點(diǎn):
完全可選的: 你在項(xiàng)目中無需重寫任何已有代碼, 就可以在一些組件中嘗試使用Hook。但是如果你不想,你不必現(xiàn)在就去學(xué)習(xí)或使用 Hook(不使用Hook也是OK的)。
100% 向后兼容的: 沒有兼容性問題, Hook 不包含任何破壞性改動(dòng)。
現(xiàn)在可用: Hook 已發(fā)布于 v16.8.0, 現(xiàn)在已經(jīng)非常穩(wěn)定, 可以放心使用。
計(jì)數(shù)器案例的對(duì)比
接下來我們通過一個(gè)計(jì)數(shù)器案例,分別使用class組件和結(jié)合了hooks的函數(shù)式組件進(jìn)行對(duì)比:
類組件的實(shí)現(xiàn)
import React, { PureComponent } from 'react' export class Counter1 extends PureComponent { constructor() { super() this.state = { counter: 10 } } changeNumber(num) { this.setState({ counter: this.state.counter + num }) } render() { const { counter } = this.state return ( <div> <h2>當(dāng)前計(jì)數(shù): {counter}</h2> <button onClick={() => this.changeNumber(-1)}>-1</button> <button onClick={() => this.changeNumber(1)}>+1</button> </div> ) } } export default Counter1
函數(shù)組件的實(shí)現(xiàn), 在下面我有使用一個(gè)hook函數(shù)useState, 先體驗(yàn)一下, 下一篇文章就會(huì)詳細(xì) 講解這個(gè)hook函數(shù)的用法
import { memo, useState } from "react" const Counter2 = memo(() => { const [ counter, setCounter ] = useState(100) return ( <div> <h2>當(dāng)前計(jì)數(shù): {counter}</h2> <button onClick={() => setCounter(counter - 1)}>-1</button> <button onClick={() => setCounter(counter + 1)}>+1</button> </div> ) }) export default Counter2
可以發(fā)現(xiàn)上面的代碼差異非常大:
函數(shù)式組件結(jié)合hooks讓整個(gè)代碼變得非常簡(jiǎn)潔
并且再也不用考慮this相關(guān)的問題;
到此這篇關(guān)于React類組件和函數(shù)組件對(duì)比-Hooks的介紹及初體驗(yàn)的文章就介紹到這了,更多相關(guān)React類組件和函數(shù)組件內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
react?hooks深拷貝后無法保留視圖狀態(tài)解決方法
這篇文章主要為大家介紹了react?hooks深拷貝后無法保留視圖狀態(tài)解決示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06React項(xiàng)目中使用Redux的?react-redux
這篇文章主要介紹了React項(xiàng)目中使用Redux的?react-redux,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-09-09React Hook useState useEffect componentD
這篇文章主要介紹了React Hook useState useEffect componentDidMount componentDidUpdate componentWillUnmount問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03TS裝飾器bindThis優(yōu)雅實(shí)現(xiàn)React類組件中this綁定
這篇文章主要為大家介紹了TS裝飾器bindThis優(yōu)雅實(shí)現(xiàn)React類組件中this綁定,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11React Hooks之使用useCallback和useMemo進(jìn)行性能優(yōu)化方式
這篇文章主要介紹了React Hooks之使用useCallback和useMemo進(jìn)行性能優(yōu)化方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-06-06