redux-saga簡介

redux-saga GitHub地址

https://github.com/yelouafi/redux-saga

redux-saga 官方文檔

文檔地址

redux-saga 簡介

一種專門用來處理異步請(qǐng)求(例如:fetch)的redux插件

使用方式

npm install --save redux-saga

當(dāng)然也支持UMD的加載方式

https://unpkg.com/redux-saga/dist/redux-saga.js
https://unpkg.com/redux-saga/dist/redux-saga.min.js

以前的方法有什么缺點(diǎn)

通過redux-thunk使用action創(chuàng)造器做一些流程控制的事情示弓。這樣可能會(huì)很難測(cè)試柄慰,但是管用。

// an action creator with thunking
function createRequest () {
  return function (dispatch, getState) {
    dispatch({ type: 'REQUEST_STUFF' })
    someApiCall(function(response) {
      // some processing
      dispatch({ type: 'RECEIVE_STUFF' })
    }
  }
}

可能在組件中還會(huì)寫到

function onHandlePress () {
  this.props.dispatch({ type: 'SHOW_WAITING_MODAL' })
  this.props.dispatch(createRequest())
}

使用redux狀態(tài)樹(和reducers)來扮演一個(gè)信號(hào)系統(tǒng)搏恤,把東西鏈接在一起。 到處都是代碼

Saga長什么樣子?

Saga其實(shí)是generator生成器函數(shù)

function *hello() {
}

對(duì)thunk的改進(jìn)

function *hello() {
  yield take('BEGIN_REQUEST')
  yield put({ type: 'SHOW_WAITING_MODAL' })
  const response = yield call(myApiFunctionThatWrapsFetch)
  yield put({ type: 'PRELOAD_IMAGES', response.images })
  yield put({ type: 'USAGE_TRACK', activity: 'API_CALL'})
  yield put({ type: 'HIDE_WAITING_MODAL' })
}

示例

[圖片上傳失敗...(image-d8de54-1527745550799)]

如果簡單的用react-redux實(shí)現(xiàn),缺點(diǎn)是:組件視圖里面會(huì)混入一些應(yīng)用邏輯代碼

class Timer extends Component {
  componentWillReceiveProps(nextProps) {
    const { state: { status: currStatus } } = this.props;
    const { state: { status: nextStatus } } = nextProps;

    if (currState === 'Stopped' && nextState === 'Running') {
      this._startTimer();
    } else if (currState === 'Running' && nextState === 'Stopped') {
      this._stopTimer();
    }
  }

  _startTimer() {
    this._intervalId = setInterval(() => {
        this.props.tick();
    }, 1000);
  }

  _stopTimer() {
    clearInterval(this._intervalId);
  }

  // ...
}

如果采用redux-thunk,我們可以這樣寫,缺點(diǎn)是:簡單地將應(yīng)用邏輯引入到了action中遏插,導(dǎo)致action過于復(fù)雜,并且不利于測(cè)試

export default {
  start: () => (
    (dispatch, getState) => {
      // This transitions state to Running
      dispatch({ type: 'START' });

      // Check every 1 second if we are still Running.
      // If so, then dispatch a `TICK`, otherwise stop
      // the timer.
      const intervalId = setInterval(() => {
        const { status } = getState();

        if (status === 'Running') {
          dispatch({ type: 'TICK' });
        } else {
          clearInterval(intervalId);
        }
      }, 1000);
    }
  )
  // ...
};

采用redux-saga纠修,就可以完美地單獨(dú)構(gòu)建應(yīng)用邏輯

function* runTimer(getState) {
  // The sagasMiddleware will start running this generator.

  // Wake up when user starts timer.
  while(yield take('START')) {
    while(true) {
      // This side effect is not run yet, so it can be treated
      // as data, making it easier to test if needed.
      yield call(wait, ONE_SECOND);

      // Check if the timer is still running.
      // If so, then dispatch a TICK.
      if (getState().status === 'Running') {
        yield put(actions.tick());
      // Otherwise, go idle until user starts the timer again.
      } else {
        break;
      }
    }
  }
}

示例源碼地址
推薦文章

常用API

初始化API

Middleware API

createSagaMiddleware(options)

用于創(chuàng)建一個(gè)中間件并連接saga和redux胳嘲。

middleware.run(saga, ...args)

啟動(dòng)saga,此方法能自動(dòng)檢測(cè)yield的執(zhí)行扣草,當(dāng)前yield返回時(shí)了牛,會(huì)執(zhí)行next(result)颜屠,result是上一個(gè)yield返回的結(jié)果。一般saga的Generator函數(shù)會(huì)寫成try/catch的形式鹰祸,當(dāng)執(zhí)行出錯(cuò)是就會(huì)執(zhí)行catch甫窟。如果saga在中途被cancel掉了,就會(huì)執(zhí)行Generator里的return蛙婴。

上層API
takeEvery(pattern, saga, ...args)

相當(dāng)于redux-thunk的執(zhí)行模式粗井,響應(yīng)匹配到的狀態(tài)多少次,執(zhí)行多少次,但是不會(huì)根據(jù)響應(yīng)的次序返回response街图。

takeLatest(pattern, saga, ...args)

如果上一次異步響應(yīng)還沒有返回浇衬,此時(shí)又接收到一次匹配的狀態(tài),則會(huì)cancel掉前一次的請(qǐng)求餐济,去執(zhí)行最新的一次請(qǐng)求耘擂。


##### 底層API
take(pattern)

相對(duì)于takeEvery,更底層絮姆,但是更實(shí)用,實(shí)例代碼如下:

function* watchAndLog() {
  yield takeEvery('*', function* logger(action) {
    const state = yield select()

    console.log('action', action)
    console.log('state after', state)
  })
}
function* loginFlow() {
  while (true) {
    yield take('LOGIN')
    // ... perform the login logic
    yield take('LOGOUT')
    // ... perform the logout logic
  }
}

使用Channels

import { take, fork, ... } from 'redux-saga/effects'

function* watchRequests() {
  while (true) {
    const {payload} = yield take('REQUEST')
    yield fork(handleRequest, payload)
  }
}

function* handleRequest(payload) { ... }

actionChannel

actionChannel用來接收redux store內(nèi)部狀態(tài)消息

import { take, actionChannel, call, ... } from 'redux-saga/effects'

function* watchRequests() {
  // 1- Create a channel for request actions
  const requestChan = yield actionChannel('REQUEST')
  while (true) {
    // 2- take from the channel
    // const requestChan = yield actionChannel('REQUEST', buffers.sliding(5))
    const {payload} = yield take(requestChan)
    // 3- Note that we're using a blocking call
    yield call(handleRequest, payload)
  }
}

function* handleRequest(payload) { ... }

eventChannel

eventChannel用來接收redux store之外的消息

import { take, put, call, cancelled } from 'redux-saga/effects'
import { eventChannel, END } from 'redux-saga'

// creates an event Channel from an interval of seconds
function countdown(seconds) { ... }

export function* saga() {
  const chan = yield call(countdown, value)
  try {    
    while (true) {
      let seconds = yield take(chan)
      console.log(`countdown: ${seconds}`)
    }
  } finally {
    if (yield cancelled()) {
      chan.close()
      console.log('countdown cancelled')
    }    
  }
}

channel

channel用在多個(gè)saga之間通信

import { channel } from 'redux-saga'
import { take, fork, ... } from 'redux-saga/effects'

function* watchRequests() {
  // create a channel to queue incoming requests
  const chan = yield call(channel)

  // create 3 worker 'threads'
  for (var i = 0; i < 3; i++) {
    yield fork(handleRequest, chan)
  }

  while (true) {
    const {payload} = yield take('REQUEST')
    yield put(chan, payload)
  }
}

function* handleRequest(chan) {
  while (true) {
    const payload = yield take(chan)
    // process the request
  }
}

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末醉冤,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子篙悯,更是在濱河造成了極大的恐慌蚁阳,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,331評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件辕近,死亡現(xiàn)場(chǎng)離奇詭異韵吨,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)移宅,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,372評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來椿疗,“玉大人漏峰,你說我怎么就攤上這事〗扉” “怎么了浅乔?”我有些...
    開封第一講書人閱讀 167,755評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長铝条。 經(jīng)常有香客問我靖苇,道長,這世上最難降的妖魔是什么班缰? 我笑而不...
    開封第一講書人閱讀 59,528評(píng)論 1 296
  • 正文 為了忘掉前任贤壁,我火速辦了婚禮,結(jié)果婚禮上埠忘,老公的妹妹穿的比我還像新娘脾拆。我一直安慰自己馒索,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,526評(píng)論 6 397
  • 文/花漫 我一把揭開白布名船。 她就那樣靜靜地躺著绰上,像睡著了一般。 火紅的嫁衣襯著肌膚如雪渠驼。 梳的紋絲不亂的頭發(fā)上蜈块,一...
    開封第一講書人閱讀 52,166評(píng)論 1 308
  • 那天,我揣著相機(jī)與錄音迷扇,去河邊找鬼疯趟。 笑死,一個(gè)胖子當(dāng)著我的面吹牛谋梭,可吹牛的內(nèi)容都是我干的信峻。 我是一名探鬼主播,決...
    沈念sama閱讀 40,768評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼瓮床,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼盹舞!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起隘庄,我...
    開封第一講書人閱讀 39,664評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤踢步,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后丑掺,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體获印,經(jīng)...
    沈念sama閱讀 46,205評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,290評(píng)論 3 340
  • 正文 我和宋清朗相戀三年街州,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了兼丰。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,435評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡唆缴,死狀恐怖鳍征,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情面徽,我是刑警寧澤艳丛,帶...
    沈念sama閱讀 36,126評(píng)論 5 349
  • 正文 年R本政府宣布,位于F島的核電站趟紊,受9級(jí)特大地震影響氮双,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜霎匈,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,804評(píng)論 3 333
  • 文/蒙蒙 一戴差、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧唧躲,春花似錦造挽、人聲如沸碱璃。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,276評(píng)論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽嵌器。三九已至,卻和暖如春谐丢,著一層夾襖步出監(jiān)牢的瞬間爽航,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,393評(píng)論 1 272
  • 我被黑心中介騙來泰國打工乾忱, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留讥珍,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,818評(píng)論 3 376
  • 正文 我出身青樓窄瘟,卻偏偏與公主長得像衷佃,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子蹄葱,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,442評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容