你們好,我是卡頌。html
你是否很討厭Hooks
調用順序的限制(Hooks
不能寫在條件語句裏)?前端
你是否遇到過在useEffect
中使用了某個state
,又忘記將其加入依賴項
,致使useEffect
回調執行時機出問題?react
怪本身粗心?怪本身很差好看文檔?數組
答應我,不要怪本身。markdown
根本緣由在於React
沒有將Hooks
實現爲響應式更新。數據結構
是實現難度很高麼?本文會用50行代碼實現無限制版Hooks
,其中涉及的知識也是Vue
、Mobx
等基於響應式更新的庫的底層原理。框架
本文的正確食用方式是收藏後用電腦看,跟着我一塊兒敲代碼(完整在線Demo
連接見文章結尾)。函數
手機黨要是看了懵逼的話不要自責,是你食用方式不對。oop
注:本文代碼來自Ryan Carniato的文章Building a Reactive Library from Scratch,老哥是
SolidJS
做者ui
首先來實現useState
:
function useState(value) {
const getter = () => value;
const setter = (newValue) => value = newValue;
return [getter, setter];
}
複製代碼
返回值數組第一項負責取值,第二項負責賦值。相比React
,咱們有個小改動:返回值的第一個參數是個函數而不是state
自己。
使用方式以下:
const [count, setCount] = useState(0);
console.log(count()); // 0
setCount(1);
console.log(count()); // 1
複製代碼
接下來實現useEffect
,包括幾個要點:
依賴的state
改變,useEffect
回調執行
不須要顯式的指定依賴項(即React
中useEffect
的第二個參數)
舉個例子:
const [count, setCount] = useState(0);
useEffect(() => {
window.title = count();
})
useEffect(() => {
console.log('沒我啥事兒')
})
複製代碼
count
變化後第一個useEffect
會執行回調(由於他內部依賴count
),可是第二個useEffect
不會執行。
前端沒有黑魔法,這裏是如何實現的呢?
答案是:訂閱發佈。
繼續用上面的例子來解釋訂閱發佈關係創建的時機:
const [count, setCount] = useState(0);
useEffect(() => {
window.title = count();
})
複製代碼
當useEffect
定義後他的回調會馬上執行一次,在其內部會執行:
window.title = count();
複製代碼
count
執行時會創建effect
與state
之間訂閱發佈的關係。
當下次執行setCount
(setter)時會通知訂閱了count
變化的useEffect
,執行其回調函數。
數據結構之間的關係如圖:
每一個useState
內部有個集合subs
,用來保存訂閱該state變化的effect
。
effect
是每一個useEffect
對應的數據結構:
const effect = {
execute,
deps: new Set()
}
複製代碼
其中:
execute
:該useEffect
的回調函數
deps
:該useEffect
依賴的state
對應subs
的集合
我知道你有點暈。看看上面的結構圖,緩緩,咱再繼續。
首先須要一個棧來保存當前正在執行的effect
。這樣當調用getter
時state
才知道應該與哪一個effect
創建聯繫。
舉個例子:
// effect1
useEffect(() => {
window.title = count();
})
// effect2
useEffect(() => {
console.log('沒我啥事兒')
})
複製代碼
count
執行時須要知道本身處在effect1
的上下文中(而不是effect2
),這樣才能與effect1
創建聯繫。
// 當前正在執行effect的棧
const effectStack = [];
複製代碼
接下來實現useEffect
,包括以下功能點:
每次useEffect
回調執行前重置依賴(回調內部state
的getter
會重建依賴關係)
回調執行時確保當前effect
處在effectStack
棧頂
回調執行後將當前effect
從棧頂彈出
代碼以下:
function useEffect(callback) {
const execute = () => {
// 重置依賴
cleanup(effect);
// 推入棧頂
effectStack.push(effect);
try {
callback();
} finally {
// 出棧
effectStack.pop();
}
}
const effect = {
execute,
deps: new Set()
}
// 馬上執行一次,創建依賴關係
execute();
}
複製代碼
cleanup
用來移除該effect
與全部他依賴的state
之間的聯繫,包括:
訂閱關係:將該effect
訂閱的全部state
變化移除
依賴關係:將該effect
依賴的全部state
移除
function cleanup(effect) {
// 將該effect訂閱的全部state變化移除
for (const dep of effect.deps) {
dep.delete(effect);
}
// 將該effect依賴的全部state移除
effect.deps.clear();
}
複製代碼
移除後,執行useEffect
回調會再逐一重建關係。
接下來改造useState
,完成創建訂閱發佈關係的邏輯,要點以下:
調用getter
時獲取當前上下文的effect
,創建關係
調用setter
時通知全部訂閱該state
變化的effect
回調執行
function useState(value) {
// 訂閱列表
const subs = new Set();
const getter = () => {
// 獲取當前上下文的effect
const effect = effectStack[effectStack.length - 1];
if (effect) {
// 創建聯繫
subscribe(effect, subs);
}
return value;
}
const setter = (nextValue) => {
value = nextValue;
// 通知全部訂閱該state變化的effect回調執行
for (const sub of [...subs]) {
sub.execute();
}
}
return [getter, setter];
}
複製代碼
subscribe
的實現,一樣包括2個關係的創建:
function subscribe(effect, subs) {
// 訂閱關係創建
subs.add(effect);
// 依賴關係創建
effect.deps.add(subs);
}
複製代碼
讓咱們來試驗下:
const [name1, setName1] = useState('KaSong');
useEffect(() => console.log('誰在那兒!', name1()))
// 打印: 誰在那兒! KaSong
setName1('KaKaSong');
// 打印: 誰在那兒! KaKaSong
複製代碼
接下來基於已有的2個hook
實現useMemo
:
function useMemo(callback) {
const [s, set] = useState();
useEffect(() => set(callback()));
return s;
}
複製代碼
這套50行的Hooks
還有個強大的隱藏特性:自動依賴跟蹤。
咱們拓展下上面的例子:
const [name1, setName1] = useState('KaSong');
const [name2, setName2] = useState('XiaoMing');
const [showAll, triggerShowAll] = useState(true);
const whoIsHere = useMemo(() => {
if (!showAll()) {
return name1();
}
return `${name1()} 和 ${name2()}`;
})
useEffect(() => console.log('誰在那兒!', whoIsHere()))
複製代碼
如今咱們有3個state
:name1
、name2
、showAll
。
whoIsHere
做爲memo
,依賴以上三個state
。
最後,當whoIsHere
變化時,會觸發useEffect
回調。
當以上代碼運行後,基於初始的3個state
,會計算出whoIsHere
,進而觸發useEffect
回調,打印:
// 打印:誰在那兒! KaSong 和 XiaoMing
複製代碼
接下來調用:
setName1('KaKaSong');
// 打印:誰在那兒! KaKaSong 和 XiaoMing
triggerShowAll(false);
// 打印:誰在那兒! KaKaSong
複製代碼
下面的事情就有趣了,當調用:
setName2('XiaoHong');
複製代碼
並無log
打印。
這是由於當triggerShowAll(false)
致使showAll state
爲false
後,whoIsHere
進入以下邏輯:
if (!showAll()) {
return name1();
}
複製代碼
因爲沒有執行name2
,因此name2
與whoIsHere
已經沒有訂閱發佈關係了!
只有當triggerShowAll(true)
後,whoIsHere
進入以下邏輯:
return `${name1()} 和 ${name2()}`;
複製代碼
此時whoIsHere
纔會從新依賴name1
與name2
。
自動的依賴跟蹤,是否是很酷~
至此,基於訂閱發佈,咱們實現了能夠自動依賴跟蹤的無限制Hooks
。
這套理念是最近幾年纔有人使用麼?
早在2010年初KnockoutJS
就用這種細粒度的方式實現響應式更新了。
不知道那時候,Steve Sanderson(KnockoutJS
做者)有沒有預見到10年後的今天,細粒度更新會在各類庫和框架中被普遍使用。
這裏是:完整在線Demo連接