Redux-Saga實現

Redux-saga

  • redux-saga是一個redux中間鍵,其經過反作用方法致力於使應用數據更容易管理,執行而且容易測試
  • 在reducer中都是純函數,但在實際開發中,咱們須要設計異步請求以及不純粹的操做(「反作用」),不像redux-thunk的侵入式,saga提供一個完整的方式,但也是結構變得複雜。
  • 工做原理以Generator函數來yield Effects

index.js

export default function createSagaMiddleware() {
    function createChannel() {
        let listener = {}
        function subscribe(actionType, cb) {
            listener[actionType]=cb;
        } 
        function publish(action) {
            if(listener[action.type]) {
                let temp = listener[action.type];
                delete listener[action.type];
                temp(action)
            }
        }
        return {subscribe, publish}
    }
    let channel = createChannel();
    function times(cb, total) {
        let count = 0;
        return function() {
            if(++count === total) {
                cb()
            }
        }
    }
    function sagaMiddleware({getState,dispatch}) {
        function run(generator) {
            let it= typeof generator === 'function' ? generator() : generator;
            function next(action) {
                let {value:effect,done}=it.next();
                if(!done){
                    if(typeof effect[Symbol.iterator] == 'function') {
                        run(effect);
                        next()
                    } else if(effect.then) {
                        effect.then(next)
                    } else {
                        switch(effect.type) {
                            case 'take':
                                channel.subscribe(effect.actionType,next);
                                break;
                            case 'put':
                                dispatch(effect.action);
                                next();
                                break;
                            case 'fork':
                                let newTask = effect.task()
                                run(newTask);
                                next(newTask);
                                break;
                            case 'call':
                                effect.fn(...effect.arg).then(next);
                                break;
                            case 'cps':
                                effect.fn(...effect.args, next);
                                break;
                            case 'all':
                                let fns=effect.fns;
                                let done=times(next, fns.length);
                                for (let i=0;i<fns.length;i++){
                                    let fn=fns[i];
                                    run(fn,done);
                                }
                                break;
                            case 'cancel':
                                effect.task.return('over');
                                break;
                            default:
                                break;
                        }
                    }
                }
            }
            next()
        }
        sagaMiddleware.run = run;
        return function(next) {
            return function(action) {
                channel.publish(action);
                next(action)
            }
        }
    } 
    return sagaMiddleware;
}
複製代碼

effects.js

export function take(actionType) {
    return {
        type:'take',
        actionType
    }
}

export function put(action) {
    return {
        type: 'put',
        action
    }
}

export function fork(task) {
    return {
        type: 'fork',
        task
    }
}

export  function* takeEvery(actionType, task) {
    yield fork(function* () {
        while (true) {
            yield take(actionType);
            yield task();
        }
    })
}

export function call(fn, ...args) {
    return {
        type: 'call',
        fn, 
        args
    }
}

const innerDelay = ms => new Promise((resolve, reject) => {
    setTimeout(() => {
        resolve()
    }, ms)
})

export function delay(...args) {
    return call(innerDelay, ...args)
}

export function cps(fn, ...args) {
    return {
        type: 'cps',
        fn,
        args
    }
}

export function all(fns) {
    return {
        type: 'all',
        fns
    }
}
複製代碼
相關文章
相關標籤/搜索