如何實(shí)現(xiàn)一個(gè)React Hooks

Hooks

Hooks是一種比較簡(jiǎn)單的方法,將state和action以及effect封裝在用戶界面中这刷。最初在React中引用狗超,現(xiàn)在已被Vue,Svelte等其他框架廣泛引用。Hooks的設(shè)計(jì)需要您對(duì)js閉包( 閉包是指某個(gè)函數(shù)能夠記住并訪問(wèn)其詞法范圍吊输,即使該函數(shù)在其詞法范圍之外執(zhí)行)的概念有充分的了解蹦魔。

useState


function useState(initialValue) {
  let _val = initialValue;
  function state() {
    return _val;
  }
  function setState(newVal) {
    _val = newVal;
  }
  return [state, setState];
}
var [foo, setFoo] = useState(0)
console.log(foo())
setFoo(1)
console.log(foo())

我們創(chuàng)建了一個(gè)和React Hooks中類似的useState,在我們函數(shù)中我們創(chuàng)建了兩個(gè)內(nèi)部函數(shù)state和setState激率。state返回了函數(shù)內(nèi)部的局部變量_val,并將使用setState設(shè)置新的值咳燕。我們借助foo和setFoo去操作了和訪問(wèn)了內(nèi)部變量_val勿决。它們保留了對(duì)useState的作用域的訪問(wèn)權(quán)限,這就叫做閉包。在React和其他框架的上下文中招盲,這看起來(lái)像狀態(tài)低缩。

讓我們將useState應(yīng)用到熟悉的環(huán)境中。我們將組成一個(gè)Counter組件!

// Example 1
function Counter() {
const [count, setCount] = useState(0)
return {
click: () => setCount(count() + 1),
render: () => console.log('render:', { count: count() })
}
}
const C = Counter()
C.render() // render: { count: 0 }
C.click()
C.render() // render: { count: 1 }

如果我們想創(chuàng)建出類似React API咆繁,我們的狀態(tài)必須是變量而不是函數(shù)讳推。

function useState(initialValue) {
var _val = initialValue
// no state() function
function setState(newVal) {
_val = newVal
}
return [_val, setState]
}
var [foo, setFoo] = useState(0)
console.log(foo)
setFoo(1)
console.log(foo)

這將是錯(cuò)誤的

我們可以u(píng)seState通過(guò)以下方法解決難題,使用module scope模式玩般。

const MyReact = (function() {
let _val;
return {
render(Component) {
const Comp = Component()
Comp.render()
return Comp
},
useState(initialValue) {
_val = _val || initialValue
function setState(newVal) {
_val = newVal
}
return [_val, setState]
}
}
})()

在這里银觅,我們選擇使用Module模式。像React一樣坏为,它跟蹤組件狀態(tài)

function Counter() {
const [count, setCount] = MyReact.useState(0)
return {
click: () => setCount(count + 1),
render: () => console.log('render:', { count })
}
}
let App
App = MyReact.render(Counter) // render: { count: 0 }
App.click()
App = MyReact.render(Counter) // render: { count: 1 }

useEffect

我們已經(jīng)介紹了useState究驴,這是第一個(gè)基本的React Hook。下一個(gè)最重要的Hooks是useEffect匀伏。不同于setState洒忧,useEffect它是異步執(zhí)行的,這意味著有更多機(jī)會(huì)遇到閉包問(wèn)題够颠。

我們可以擴(kuò)展到目前為止已經(jīng)建立的React微模型熙侍,以包括以下內(nèi)容:

// Example 3
const MyReact = (function() {
let _val, _deps // hold our state and dependencies in scope
return {
render(Component) {
const Comp = Component()
Comp.render()
return Comp
},
useEffect(callback, depArray) {
const hasNoDeps = !depArray
const hasChangedDeps = _deps ? !depArray.every((el, i) => el === _deps[i]) : true
if (hasNoDeps || hasChangedDeps) {
callback()
_deps = depArray
}
},
useState(initialValue) {
_val = _val || initialValue
function setState(newVal) {
_val = newVal
}
return [_val, setState]
}
}
})()

// usage
function Counter() {
const [count, setCount] = MyReact.useState(0)
MyReact.useEffect(() => {
console.log('effect', count)
}, [count])
return {
click: () => setCount(count + 1),
noop: () => setCount(count),
render: () => console.log('render', { count })
}
}
let App
App = MyReact.render(Counter)
// effect 0
// render {count: 0}
App.click()
App = MyReact.render(Counter)
// effect 1
// render {count: 1}
App.noop()
App = MyReact.render(Counter)
// // no effect run
// render {count: 1}
App.click()
App = MyReact.render(Counter)
// effect 2
// render {count: 2}

為了跟蹤依賴關(guān)系(因?yàn)閡seEffect依賴關(guān)系發(fā)生更改后會(huì)重新運(yùn)行),我們引入了另一個(gè)變量track _deps履磨。

我們對(duì)useState和useEffect功能進(jìn)行了很好的復(fù)制蛉抓,但兩者均實(shí)現(xiàn)不好。

// Example 4
const MyReact = (function() {
let hooks = [],
currentHook = 0 // array of hooks, and an iterator!
return {
render(Component) {
const Comp = Component() // run effects
Comp.render()
currentHook = 0 // reset for next render
return Comp
},
useEffect(callback, depArray) {
const hasNoDeps = !depArray
const deps = hooks[currentHook] // type: array | undefined
const hasChangedDeps = deps ? !depArray.every((el, i) => el === deps[i]) : true
if (hasNoDeps || hasChangedDeps) {
callback()
hooks[currentHook] = depArray
}
currentHook++ // done with this hook
},
useState(initialValue) {
hooks[currentHook] = hooks[currentHook] || initialValue // type: any
const setStateHookIndex = currentHook // for setState's closure!
const setState = newState => (hooks[setStateHookIndex] = newState)
return [hooks[currentHook++], setState]
}
}
})()

// Example 4 continued - in usage
function Counter() {
const [count, setCount] = MyReact.useState(0)
const [text, setText] = MyReact.useState('foo') // 2nd state hook!
MyReact.useEffect(() => {
console.log('effect', count, text)
}, [count, text])
return {
click: () => setCount(count + 1),
type: txt => setText(txt),
noop: () => setCount(count),
render: () => console.log('render', { count, text })
}
}
let App
App = MyReact.render(Counter)
// effect 0 foo
// render {count: 0, text: 'foo'}
App.click()
App = MyReact.render(Counter)
// effect 1 foo
// render {count: 1, text: 'foo'}
App.type('bar')
App = MyReact.render(Counter)
// effect 1 bar
// render {count: 1, text: 'bar'}
App.noop()
App = MyReact.render(Counter)
// // no effect run
// render {count: 1, text: 'bar'}
App.click()
App = MyReact.render(Counter)
// effect 2 bar
// render {count: 2, text: 'bar'}

自定義的Hooks:

// Example 4, revisited
function Component() {
const [text, setText] = useSplitURL('www.netlify.com')
return {
type: txt => setText(txt),
render: () => console.log({ text })
}
}
function useSplitURL(str) {
const [text, setText] = MyReact.useState(str)
const masked = text.split('.')
return [masked, setText]
}
let App
App = MyReact.render(Component)
// { text: [ 'www', 'netlify', 'com' ] }
App.type('www.reactjs.org')
App = MyReact.render(Component)
// { text: [ 'www', 'reactjs', 'org' ] }}

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末蹬耘,一起剝皮案震驚了整個(gè)濱河市芝雪,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌综苔,老刑警劉巖惩系,帶你破解...
    沈念sama閱讀 210,978評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異如筛,居然都是意外死亡堡牡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,954評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門杨刨,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)晤柄,“玉大人,你說(shuō)我怎么就攤上這事妖胀〗婢保” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,623評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵赚抡,是天一觀的道長(zhǎng)爬坑。 經(jīng)常有香客問(wèn)我,道長(zhǎng)涂臣,這世上最難降的妖魔是什么盾计? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,324評(píng)論 1 282
  • 正文 為了忘掉前任售担,我火速辦了婚禮,結(jié)果婚禮上署辉,老公的妹妹穿的比我還像新娘族铆。我一直安慰自己,他們只是感情好哭尝,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,390評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布哥攘。 她就那樣靜靜地躺著,像睡著了一般材鹦。 火紅的嫁衣襯著肌膚如雪献丑。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,741評(píng)論 1 289
  • 那天侠姑,我揣著相機(jī)與錄音创橄,去河邊找鬼。 笑死莽红,一個(gè)胖子當(dāng)著我的面吹牛妥畏,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播安吁,決...
    沈念sama閱讀 38,892評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼醉蚁,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了鬼店?” 一聲冷哼從身側(cè)響起网棍,我...
    開(kāi)封第一講書(shū)人閱讀 37,655評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎妇智,沒(méi)想到半個(gè)月后滥玷,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,104評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡巍棱,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,451評(píng)論 2 325
  • 正文 我和宋清朗相戀三年惑畴,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片航徙。...
    茶點(diǎn)故事閱讀 38,569評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡如贷,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出到踏,到底是詐尸還是另有隱情杠袱,我是刑警寧澤,帶...
    沈念sama閱讀 34,254評(píng)論 4 328
  • 正文 年R本政府宣布窝稿,位于F島的核電站楣富,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏讹躯。R本人自食惡果不足惜菩彬,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,834評(píng)論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望潮梯。 院中可真熱鬧骗灶,春花似錦、人聲如沸秉馏。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,725評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)萝究。三九已至免都,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間帆竹,已是汗流浹背绕娘。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,950評(píng)論 1 264
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留栽连,地道東北人险领。 一個(gè)月前我還...
    沈念sama閱讀 46,260評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像秒紧,于是被迫代替她去往敵國(guó)和親绢陌。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,446評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容

  • Hooks 是React的一次革命性升級(jí)熔恢,本文將對(duì)其優(yōu)勢(shì)和API進(jìn)行比較全面的解析 為什么要有hooks 在沒(méi)有h...
    smartzheng閱讀 931評(píng)論 0 5
  • React是現(xiàn)在最流行的前端框架之一脐湾,它的輕量化,組件化叙淌,單向數(shù)據(jù)流等特性把前端引入了一個(gè)新的高度秤掌,現(xiàn)在它又引入的...
    老鼠AI大米_Java全棧閱讀 5,775評(píng)論 0 26
  • 原文地址React Hooks實(shí)例教學(xué) 目前,Hooks 應(yīng)該是 React 中最火的概念了鹰霍,在閱讀這篇文章之前机杜,...
    DC_er閱讀 15,772評(píng)論 4 14
  • 最新在學(xué)ReactHooks這個(gè)新特性椒拗,把學(xué)習(xí)筆記記下來(lái),供大家分享获黔。 原先的函數(shù)組件是沒(méi)有生命周期函數(shù)的蚀苛,這樣在...
    番茄_tomatoMan閱讀 1,737評(píng)論 0 0
  • React Hooks Hook 是能讓你在函數(shù)組件中“鉤入” React 特性的函數(shù)。 State Hook u...
    梁坤同學(xué)閱讀 340評(píng)論 0 0