網站首頁 編程語言 正文
1.首先創建一個store
沙箱鏈接
根目錄創建一個store文件夾,下面創建一個index.js
import { createStore } from '../my-redux'
// 書寫reducer函數
function reducer(state = 0, action) {
switch (action.type) {
case "add":
return state + 1;
case "inc":
return state - 1;
default:
return state;
}
}
// 使用createStore(reducer)創建store對象并且導出
const state = createStore(reducer);
export default state;
結合上面的代碼分析
- 創建store是redux庫的createStore函數接收一個reducer函數進行創建。
import { createStore } from '../my-redux'
- 先手寫一個簡單的reducer函數
// 書寫reducer函數
狀態值默認為0
function reducer(state = 0, action) {
switch (action.type) {
case "add":
return state + 1;
case "inc":
return state - 1;
default:
return state;
}
}
- 將創建的store導出
// 使用createStore(reducer)創建store對象并且導出
const state = createStore(reducer);
export default state;
2.其次創建一個my-redux
- 將所有的函數都導入index.js
import createStore from './createStore'
export {
createStore
}
- 創建一個createStore.js
export default function createStore(reducer) {
let currentState; // 當前state值
let currentListeners = []; // store訂閱要執行的函數儲存數組
// 獲得當前state值
function getState() {
return currentState;
}
// 更新state
function dispatch(action) {
// 傳入action 調用reducer更新state值
currentState = reducer(currentState, action)
// 遍歷調用訂閱的函數
currentListeners.forEach((listener) => listener());
}
// 將訂閱事件儲存到currentListeners數組,并返回unsubscribe 函數來取消訂閱
function subscribe(listener) {
currentListeners.push(listener);
// unsubscribe
return () => {
const index = currentListeners.indexOf(listener);
currentListeners.splice(index, 1);
};
}
dispatch({ type: "" }); // 自動dispatch一次,保證剛開始的currentState值等于state初始值
// 返回store對象
return {
getState,
dispatch,
subscribe,
}
}
可以根據上面給出的代碼步步分析:
①明確createStore接收一個reducer函數作為參數。
②createStore函數返回的是一個store對象,store對象包含getState,dispatch,subscribe等方法。
- 逐步書寫store上的方法
書寫getState()方法
返回值:當前狀態值
// 獲得當前state值
function getState() {
return currentState;
}
書寫dispatch方法
接受參數:action。
dispatch方法,做的事情就是:①調用reducer函數更新state。②調用store訂閱的事件函數。
currentState是當前狀態值,currentListeners是儲存訂閱事件函數的數組。
// 更新state
function dispatch(action) {
// 傳入action 調用reducer更新state值
currentState = reducer(currentState, action)
// 遍歷調用訂閱的函數
currentListeners.forEach((listener) => listener());
}
書寫subscribe方法
接受參數:一個函數 返回值:一個函數,用于取消訂閱unsubscribe
// 將訂閱事件儲存到currentListeners數組,并返回unsubscribe 函數來取消訂閱
function subscribe(listener) {
currentListeners.push(listener);
// unsubscribe
return () => {
const index = currentListeners.indexOf(listener);
currentListeners.splice(index, 1); // 刪除函數
};
}
返回store對象
// 返回store對象
return {
getState,
dispatch,
subscribe,
}
特別注意:
初始進入createStore函數的時候,需要通過dipatch方法傳入一個獨一無二的action(reducer函數默認返回state)來獲取初始的store賦值給currentStore。
可以結合下面reducer的default項和createStore方法調用的dispatch來理解
dispatch({ type: "" }); // 自動dispatch一次,保證剛開始的currentState值等于state初始值
// 書寫reducer函數
function reducer(state = 0, action) {
switch (action.type) {
case "add":
return state + 1;
case "inc":
return state - 1;
default:
return state;
}
}
這樣我們的createStore函數就已經完成了。那接下來就是檢查我們寫的這玩意是否起作用沒有了。
3.創建一個Test組件進行檢測。
老規矩先拋全部代碼
import React, { Component } from 'react'
import store from './store' // 引入store對象
export default class Test extends Component {
// 組件掛載之后訂閱forceUpdate函數,進行強制更新
componentDidMount() {
this.unsubscribe = store.subscribe(() => {
this.forceUpdate()
})
}
// 組件卸載后取消訂閱
componentWillUnmount() {
this.unsubscribe()
}
// handleclick事件函數
add = () => {
store.dispatch({ type: 'add' });
console.log(store.getState());
}
inc = () => {
store.dispatch({ type: 'inc' });
console.log(store.getState());
}
render() {
return (
<div>
{/* 獲取store狀態值 */}
<div>{store.getState()}</div>
<button onClick={this.add}>+</button>
<button onClick={this.inc}>-</button>
</div>
)
}
}
1. 將Test組件記得引入App根組件
import Test from './Test';
function App() {
return (
<div className="App">
<Test />
</div>
);
}
export default App;
2. 將store引入Test組件
import React, { Component } from 'react'
import store from './store' // 引入store對象
3. 創建一個類組件,并且使用store.getState()獲得狀態值
<div>
{/* 獲取store狀態值 */}
<div>{store.getState()}</div>
<button onClick={this.add}>+</button>
<button onClick={this.inc}>-</button>
</div>
4. 書寫對應的點擊按鈕
// handleclick事件函數
add = () => {
store.dispatch({ type: 'add' });
console.log(store.getState());
}
inc = () => {
store.dispatch({ type: 'inc' });
console.log(store.getState());
}
<div>
{/* 獲取store狀態值 */}
<div>{store.getState()}</div>
<button onClick={this.add}>+</button>
<button onClick={this.inc}>-</button>
</div>
這樣是不是就可以實現了呢?哈哈哈,傻瓜,你是不是猛點鼠標,數字還是0呢?
當然,這里我們只是更新了store,但是并沒有更新組件,狀態的改變可以導致組件更新,但是store又不是Test組件的狀態。
這里我們使用一個生命周期函數componentDidMount和store的訂閱函數進行更新組件的目的,使用componentWillUnmount和store的取消訂閱函數(訂閱函數的返回值)。
// 組件掛載之后訂閱forceUpdate函數,進行強制更新
componentDidMount() {
this.unsubscribe = store.subscribe(() => {
this.forceUpdate()
})
}
// 組件卸載后取消訂閱
componentWillUnmount() {
this.unsubscribe()
}
好了。這里我們就真實實現了一個簡單的createStore函數來創建store。
原文鏈接:https://juejin.cn/post/7156891589422514189
相關推薦
- 2022-05-18 C語言模擬內存函數分析之mencpy與memmove_C 語言
- 2022-10-20 三種Golang數組拷貝方式及性能分析詳解_Golang
- 2022-07-02 less,sass,scss的關系與區別
- 2022-06-08 Flutter集成高德地圖并添加自定義Maker的實踐_Android
- 2022-12-13 C++實現defer聲明方法詳解_C 語言
- 2022-03-28 c語言for、while和do-while循環之間的區別_C 語言
- 2022-09-17 Golang文件讀寫操作詳情_Golang
- 2022-04-21 詳解前端到底可以用nginx做什么_nginx
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支