Redux和@reduxjs/toolkit的使用

1. 简介:

Redux 是一种用于管理应用程序状态的 JavaScript 库。它是一个可预测的状态容器,可以用于编写
可维护和可扩展的应用程序。
@reduxjs/toolkit 是一个官方提供的 Redux 工具包,它可以帮助简化 Redux 应用程序的开发,并
提供常用的 Redux 原生方法,例如创建 Redux store、定义 reducer、处理异步操作等。

2. 用法(同步)

首先,我们需要安装 @reduxjs/toolkit 和 react-redux

npm install @reduxjs/toolkit react-redux -S
1> 引入Provider,对根目录进行一个包装
import { Provider } from 'react-redux';
import store from './store/index.js';
const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(
    <Provider store={srore}>
        <App />
    </Provider>
);
2> 编写store文件,利用@reduxjs/toolkit工具的configureStore创建store
import { configureStore } from '@reduxjs/toolkit';
import userReducer from './reducers/user';
const store = configureStore({
    reducer: {
        user: userReducer
    }
});
export default store;
3> 编写store下的reducers/user文件
import { createSlice } from '@reduxjs/toolkit';
let userSlice = createSlice({
    name: 'user',
    initialState: {
        isLogin: false,
        loginCount: 0,
	test: ''
    },
    reducers: {
        addLoginCount(state) {
            state.loginCount++;
        },
        addLoginCountByNum(state, action) {
            state.loginCount += action.payload;
        }
    },
    extraReducers(builder) { // 异步的处理
        // 当getTest fulfilled状态时,处理
        builder.addCase(getTest.fulfilled, (state, { payload }) => {
            // 上述方法返回的结果payload
            state.test = payload;
        })
    }
});
// 外部组件来使用的action
export const { addLoginCount, addLoginCountByNum } = userSlice.actions;
// 默认导出是所有的reducer 供store注册这些方法
export default userSlice.reducer;
4> 使用store
import './App.css';
import { useState, memo } from 'react';
// 取出行为
import { addLoginCount, addLoginCountByNum } from './store/reducers/user';
import { useDispatch, useSelector } from 'react-redux';

// 在子组件中使用
// 这里的memo保证props发生变化时,才更新
const Son = memo(() => {
    console.log('子组件更新.....')
    const user = useSelector(state => state.user);
    return <h1>我是子组件{user.loginCount}</h1>
})
function App() {
    const user = useSelector(state => state.user);
    const dispath = useDispatch();
    const [num, setNum] = useState(1);
    return (
        <div className="App">
            {user.isLogin ? 'TRUE' : 'FALSE'} || {user.loginCount} || {num}
            <button onClick={e => dispath(addLoginCount())}>更改loginCount+1</button>
            <button onClick={e => dispath(addLoginCountByNum(10))}>更改loginCount</button>
            <button onClick={e => setNum(num + 1)}>变更父组件</button>
            <Son />
        </div>
    );
}
export default App;

memo用法补充:
就如上边代码,在没有用memo包装前,每次修改num的值时,子组件都会随着更新。用memo包装后,组件Son不会随num值的改变而更新。但如果num的值传给了Son组件(<Son num={num}/>),Son组件也会跟随num的值的修改而更新了。大幅度使用会造成缓存越来越大,针对组件大一点且使用频率不是很高的情况下去使用。

以上是同步的用法,下边记录下异步的用法:

3. 用发(异步)

1> 首先就是就是借助@reduxjs/toolkit工具库的createAsyncThunk方法,在reducers/user文件中新增一个getTest方法模拟获取数据如下:
export const getTest = createAsyncThunk('user/getTest', async () => {
    let res = await fetch('http://localhost:3001/');
    let text = await res.text();
    console.log('text::', text)
    return text;
});

说明:上边用的fetch是ES6里新增的一个发请求的api。

2>在reducers统计中添加extraReducers的回调,其接收一个builder的参数,通过builder的addCase方法,可以根据上边定义的getTest方法的状态 (这里也就是Promise的三种状态pending/rejection/fulfilled) 去处理数据,如上边代码中的:
extraReducers(builder) { // 异步的处理
        // 当getTest fulfilled状态时,处理
        builder.addCase(getTest.fulfilled, (state, { payload }) => {
            // 上述方法返回的结果payload
            state.test = payload;
        })
}
3>使用
function App() {
    const user = useSelector(state => state.user);
    const dispath = useDispatch();
    useEffect(() => {
        dispath(getTest());
    }, [])
    return (
        <div className="App">
            <pre>
                {user.test}
            </pre>
            {user.isLogin ? 'TRUE' : 'FALSE'}
        </div>
    );
}
posted @ 2024-01-27 15:45  lvkehao  阅读(468)  评论(0编辑  收藏  举报