使用store來優(yōu)化React組件的方法
在使用 React 編寫組件的時(shí)候,我們常常會碰到兩個(gè)不同的組件之間需要共享狀態(tài)情況,而通常的做法就是提升狀態(tài)到父組件。但是這樣做會有一個(gè)問題,就是盡管只有兩個(gè)組件需要這個(gè)狀態(tài),但是因?yàn)榘褷顟B(tài)提到了父組件,那么在狀態(tài)變化的時(shí)候,父組件以及其下面的所有子組件都會重新 render,如果你的父組件比較復(fù)雜,包含了其他很多子組件的話,就有可能引起性能問題。
Redux 通過把狀態(tài)放在全局的 store 里,然后組件去訂閱各自需要的狀態(tài),當(dāng)狀態(tài)發(fā)生變化的時(shí)候,只有那些訂閱的狀態(tài)發(fā)生變化的組件才重新 render,這樣就避免了上面說的提升狀態(tài)所帶來的副作用。但是,當(dāng)我們在寫一個(gè) React 組件庫的時(shí)候,redux 加 react-redux 的組合可能就有點(diǎn)太重了。所以我們可以自己寫一個(gè)簡單的 store,來實(shí)現(xiàn)類似 Redux 的訂閱模式。
參考 Redux 的實(shí)現(xiàn)來寫一個(gè)簡版的 createStore:
function createStore(initialState) {
let state = initialState;
const listeners = [];
function setState(partial) {
state = {
...state,
...partial,
};
for (let i = 0; i < listeners.length; i++) {
listeners[i]();
}
}
function getState() {
return state;
}
function subscribe(listener) {
listeners.push(listener);
return function unsubscribe() {
const index = listeners.indexOf(listener);
listeners.splice(index, 1);
};
}
return {
setState,
getState,
subscribe,
};
}
我們的 createStore 非常簡單,算上空行也只有 33 行,總共暴露了 3 個(gè)方法,沒有 Redux 里的 dispatch 和 reducer,直接通過 setState 方法改變狀態(tài)。下面我們來用它一個(gè)計(jì)數(shù)器的例子。
class Counter extends React.Component {
constructor(props) {
super(props);
// 初始化 store
this.store = createStore({
count: 0,
});
}
render() {
return (
<div>
<Buttons store={store} />
<Result store={store} />
</div>
)
}
}
class Buttons extends React.Component {
handleClick = (step) => () => {
const { store } = this.props;
const { count } = store.getState();
store.setState({ count: count + step });
}
render() {
return (
<div>
<button onClick={this.handleClick(1)}>+</button>
<button onClick={this.handleClick(1)}>-</button>
</div>
);
}
}
class Result extends React.Component {
constructor(props) {
super(props);
this.state = {
count: props.store.getState().count,
};
}
componentDidMount() {
this.props.store.subscribe(() => {
const { count } = this.props.store.getState();
if (count !== this.state.count) {
this.setState({ count });
}
});
}
render() {
return (
<div>{this.state.count}</div>
);
};
}
例子中 Buttons 里通過 store.setState 來改變 store 中的狀態(tài),并不會引起整個(gè) Counter 的重新 render,但是因?yàn)?Result 中訂閱了 store 的變化,所以當(dāng) count 有變化的時(shí)候就可以通過改變自己組件內(nèi)的狀態(tài)來重新 render,這樣就巧妙地避免了不必須要的 render。
最后,上面的 createStore 雖然只有幾十行代碼,我還是把它寫成了一個(gè)叫 mini-store 庫放在 GitHub 上,并且提供了類似 Redux 的 Provider 和 connect 方法,總共加起來也就 100 多行代碼。如果你也在寫 React 組件庫,需要管理一個(gè)復(fù)雜組件的狀態(tài),不妨試試這個(gè)優(yōu)化方式。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
useEffect?返回函數(shù)執(zhí)行過程源碼解析
這篇文章主要為大家介紹了useEffect?返回函數(shù)執(zhí)行過程源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04
詳解React-Native全球化多語言切換工具庫react-native-i18n
這篇文章主要介紹了詳解React-Native全球化語言切換工具庫react-native-i18n,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-11-11
解決React報(bào)錯(cuò)Rendered more hooks than during
這篇文章主要為大家介紹了React報(bào)錯(cuò)Rendered more hooks than during the previous render解決方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12
使用Ant Design Anchor組件的一個(gè)坑及解決
這篇文章主要介紹了使用Ant Design Anchor組件的一個(gè)坑及解決,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-04-04
在react項(xiàng)目中使用antd的form組件,動態(tài)設(shè)置input框的值
這篇文章主要介紹了在react項(xiàng)目中使用antd的form組件,動態(tài)設(shè)置input框的值,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-10-10
React實(shí)現(xiàn)全局組件的Toast輕提示效果
這篇文章主要介紹了React實(shí)現(xiàn)全局組件的Toast輕提示效果,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-09-09
React useMemo與useCallabck有什么區(qū)別
useCallback和useMemo是一樣的東西,只是入?yún)⒂兴煌?,useCallback緩存的是回調(diào)函數(shù),如果依賴項(xiàng)沒有更新,就會使用緩存的回調(diào)函數(shù);useMemo緩存的是回調(diào)函數(shù)的return,如果依賴項(xiàng)沒有更新,就會使用緩存的return2022-12-12
react 權(quán)限樹形結(jié)構(gòu)實(shí)現(xiàn)代碼
這篇文章主要介紹了react 權(quán)限樹形結(jié)構(gòu)實(shí)現(xiàn)代碼,項(xiàng)目背景react + ant design,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧2024-05-05

