美文网首页前端修仙之路
小技巧:使用useContext和useReducer构建小型r

小技巧:使用useContext和useReducer构建小型r

作者: ryderdu | 来源:发表于2019-05-29 18:45 被阅读0次

    对于两个子组件间的通信,相信我们开发中并不少见,基于我们都不怎么喜欢redux的坚持…...其实我一直在用团队老大此前写的globalBus持续真香,我在上一篇文章也有讲到那玩意的原理和实现,但是老大说现在可以不用啦,新版hooks两个api联用,更香,于是我简单的实践了一番,发现…真香。

    首先你需要在两个子组件之上的一层架设你的context chilFirst和ChildSecond是我建立的两个组件,我们将会从first组件发dipatch,在second组件展示变化的数据,这个场景开发中经常使用。

    首先我们要建立context, 我们需要一个初始状态和一个变更state的状态管理器,咦,这怎么这么像…..

    // Context.js
    export const defaultState = {
        value: 0
    }
    
    export function reducer(state, action) {
        switch(action.type) {
            case 'ADD_NUM':
                return { ...state, value: state.value + 1 };
            case 'REDUCE_NUM':
                return { ...state, value: state.value - 1 };
            default: 
                throw new Error();
        }
    }
    

    Content.js ,这里我们需要显性声明Context.Provider 把数据传给包裹组件,这里轮到强大的useReducer出场了,按照官方要求,你需要把reducer(我们此前定义)和默认状态传入https://reactjs.org/docs/hooks-reference.html#usereducer,(这里小小利用hooks组件的状态,当我们向上传递修改后,其包裹的组件的引用props更新也会重新获得新的props)

    // Content.js
    import React, { useReducer, createContext } from 'react'
    import { ChildFirst } from './ChildFirst'
    import { ChildSecond } from './ChildSecond'
    import { reducer, defaultState } from './reducer'
    
    export const Context = createContext(null)
    
    export function Content() {
        const [state, dispatch] = useReducer(reducer, defaultState)
    
        return (
            <Context.Provider value={{state, dispatch: dispatch}}>
                <ChildFirst/>
                <ChildSecond/>
            </Context.Provider>
        )
    } 
    

    组件First ,在这个组件我们dispatch发事件试试

    // ChildFirst.js
    import React, {useContext} from 'react'
    import {Context} from './content'
    
    export function ChildFirst() {
        const AppContext = useContext(Context)
    
        return (
            <div>
                <button onClick={ 
                     () => {
                    AppContext.dispatch({
                        type: "ADD_NUM",
                        payload: {}
                      })
                    }
                }>addNum</button>
                <button onClick={
                    () => {
                    AppContext.dispatch({
                        type: "REDUCE_NUM",
                        payload: {}
                        })  
                    }
                }>reduceNum</button>
            </div>
        )
    } 
    

    组件Second

    // ChildSecond.js
    import React, {useContext} from 'react'
    import {Context} from './content'
    
    export function ChildSecond() {
        const AppContext = useContext(Context)
    
        return (
            <div>
                {AppContext.state.value + 's'}
            </div>
        )
    } 
    
    DEMO

    我们发现在局部地区架设局部组件需要公用的context,不仅有利于我们组织结构的解耦,也可以避免我们一开始就要引入redux这种全局状态库或者中途引入带来的成本。本身作为一个应用需要公共管理的状态其实并不多,再加之团队开发素质不一管理不善的话,就很容易滥用redux,极大拖慢了整个应用的速度,所以学会这个技巧,可以优化组件间的状态管理和应用的轻便性能。

    相关文章

      网友评论

        本文标题:小技巧:使用useContext和useReducer构建小型r

        本文链接:https://www.haomeiwen.com/subject/wilutctx.html