useReducer+useContext≈Redux
思路
- 建立全局的Context
- 建立全局的Reducer
- 將全局useReducer返回的state和dispatch傳遞給全局Context.Provider的value中
- 用全局構建好的帶有Context的組件包裹應用根組件
編碼(暫且叫Reduxx)
/**
* 文件名 Reduxx.js
*/
import React, { useReducer, createContext } from "react";
// 1. 建立全局的Context
const Context = createContext();
export default Context;
// 2. 建立全局的Reducer
const initState = {};
const reducer = (state, action) => {
switch (action.type) {
case "setState":
return { ...state, ...action.payload };
default:
return state;
}
};
// 3. 將全局useReducer返回的state和dispatch傳遞給全局Context.Provider的value中
export const Provider = ({ children }) => {
const [state, dispatch] = useReducer(reducer, initState);
return (
<Context.Provider value={{ state, dispatch }}>{children}</Context.Provider>
);
};
用Reduxx包裹應用根組件
import { Provider } from "./Reduxx";
import ReduxDemo from "./components/ReduxDemo";
export default function App() {
return (
<Router>
<Provider>
<div className="App">
<h3>Hooks 學習</h3>
<ul>
<li>
<Link to="/">useState</Link>
</li>
<li>
<Link to="/reduxDeom">useReducer+useContext≈Redux</Link>
</li>
</ul>
<div className="show">
<Route path="/" exact render={() => <div>useState so easy</div>} />
<Route path="/reduxDeom" exact component={ReduxDemo} />
</div>
</div>
</Provider>
</Router>
);
}
使用
/**
* ReduxDemo.js
**/
import React, { useContext } from "react";
import Context from "../Reduxx";
export default () => {
const { state, dispatch } = useContext(Context);
return (
<div>
<h3>useReducer+useContext≈Redux</h3>
<div>{JSON.stringify(state)}</div>
<button
onClick={() => {
dispatch({ type: "setState", payload: { time: Date.now() } });
}}
>
set Time
</button>
<button
onClick={() => {
dispatch({ type: "setState", payload: { random: Math.random() } });
}}
>
set Random
</button>
</div>
);
};