react hooks之useReducer+useContext實現redux簡易版

useReducer+useContext≈Redux

思路

  1. 建立全局的Context
  2. 建立全局的Reducer
  3. 將全局useReducer返回的state和dispatch傳遞給全局Context.Provider的value中
  4. 用全局構建好的帶有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>
  );
};

效果演示

完整代碼

相關文章
相關標籤/搜索