この記事では主に redux-saga の最初の導入と使用方法を紹介し、参考にしていきます。
redux-saga は Redux アプリケーションの非同期操作を管理するミドルウェアであり、その機能は redux-thunk + async/await に似ており、Saga を作成することですべての非同期操作ロジックを 1 か所に保存します。
redux-saga エフェクト
redux-saga エフェクトは、saga ミドルウェアによって実行されるいくつかの命令を含むプレーン テキストの JavaScript オブジェクトです。これらの命令によって実行される操作には、次の 3 つのタイプが含まれます:
非同期呼び出しの開始 (Ajax リクエストの送信など)
ストアを更新する他のアクションの開始
他の Saga の呼び出し
Effects には、非同期 API リファレンスに含まれる多くの命令が含まれています。redux-saga の機能は、次のようなテストに便利です。処理のためにサガに集中しています
watch/worker (listening->execution) 作業フォーム
はジェネレータとして実装されています
assert.deepEqual(iterator.next().value, call(Api.fetch, '/products'))
// user.js import request from 'axios'; // define constants // define initial state // export default reducer export const loadUserData = (uid) => async (dispatch) => { try { dispatch({ type: USERDATA_REQUEST }); let { data } = await request.get(`/users/${uid}`); dispatch({ type: USERDATA_SUCCESS, data }); } catch(error) { dispatch({ type: USERDATA_ERROR, error }); } }
redux-saga
すべての非同期ロジックは saga.js に書き込むことができます:import request from 'axios'; import { loadUserData } from './user'; export const login = (user, pass) => async (dispatch) => { try { dispatch({ type: LOGIN_REQUEST }); let { data } = await request.post('/login', { user, pass }); await dispatch(loadUserData(data.uid)); dispatch({ type: LOGIN_SUCCESS, data }); } catch(error) { dispatch({ type: LOGIN_ERROR, error }); } }
take の使用 take と takeEvery はどちらも特定のアクションを監視しますが、それらの機能は一貫性がありません。 takeEvery はアクションが実行されるたびに応答します。がトリガーされ、実行フローが take ステートメントに到達した場合にのみ take が応答します。 takeEvery はアクションをリッスンし、対応する処理関数を実行するだけであり、アクションがいつ実行されるか、および呼び出されたタスクがいつ呼び出されるかを制御することはできず、いつ停止するかを制御することもできません。 listen は、アクションが一致するたびに何度でも呼び出すことができます。ただし、Take は、ジェネレーター関数のアクションと応答後の後続の操作にいつ応答するかを決定できます。 たとえば、すべてのタイプのアクション トリガーを監視するときにロガー操作を実行するには、takeEvery を使用して次を実装します:
export function* loginSaga() { while(true) { const { user, pass } = yield take(LOGIN_REQUEST) //等待 Store 上指定的 action LOGIN_REQUEST try { let { data } = yield call(loginRequest, { user, pass }); //阻塞,请求后台数据 yield fork(loadUserData, data.uid); //非阻塞执行loadUserData yield put({ type: LOGIN_SUCCESS, data }); //发起一个action,类似于dispatch } catch(error) { yield put({ type: LOGIN_ERROR, error }); } } } export function* loadUserData(uid) { try { yield put({ type: USERDATA_REQUEST }); let { data } = yield call(userRequest, `/users/${uid}`); yield put({ type: USERDATA_SUCCESS, data }); } catch(error) { yield put({ type: USERDATA_ERROR, error }); } }
takeEvery を使用して次を実装します:
import { takeEvery } from 'redux-saga'
function* watchAndLog(getState) {
yield* takeEvery('*', function* logger(action) {
//do some logger operation //在回调函数体内
})
}
呼び出し操作は、ジェネレーターの場合、非同期操作を開始するために使用されます。ジェネレーターの呼び出しが終了するまでは、他の操作を実行したり処理したりすることはできません。ただし、fork は非ブロック操作です。fork がタスクをモバイル化すると、その時点で、実行フローは結果が返されるのを待たずに実行を継続できます。
たとえば、次のログイン シナリオ:
import { take } from 'redux-saga/effects' function* watchAndLog(getState) { while(true) { const action = yield take('*') //do some logger operation //与 take 并行 }) }
複数のタスクを同時に実行してください
特定のシナリオに遭遇し、複数のタスクを同時に実行する必要がある場合は、次のような場合があります。ユーザーデータと製品データをリクエストする場合、次のメソッドを使用する必要があります:function* loginFlow() { while(true) { const {user, password} = yield take('LOGIN_REQUEST') const token = yield call(authorize, user, password) if(token) { yield call(Api.storeItem({token})) yield take('LOGOUT') yield call(Api.clearItem('token')) } } }
import { call } from 'redux-saga/effects' //同步执行 const [users, products] = yield [ call(fetch, '/users'), call(fetch, '/products') ] //而不是 //顺序执行 const users = yield call(fetch, '/users'), products = yield call(fetch, '/products')
const sagaMiddleware = createSagaMiddleware({sagaMonitor}) const store = createStore( reducer, applyMiddleware(sagaMiddleware) ) sagaMiddleware.run(rootSaga)
这段逻辑主要是执行了 sagaMiddleware(),该函数里面将 runSaga 赋值给 sagaMiddleware.run 并执行,最后返回 middleware。 接着看 runSaga() 的逻辑:
export function runSaga(options, saga, ...args) { ... const task = proc( iterator, channel, wrapSagaDispatch(dispatch), getState, context, { sagaMonitor, logger, onError, middleware }, effectId, saga.name, ) if (sagaMonitor) { sagaMonitor.effectResolved(effectId, task) } return task }
这个函数里定义了返回了一个 task 对象,该 task 是由 proc 产生的,移步 proc.js:
export default function proc( iterator, stdChannel, dispatch = noop, getState = noop, parentContext = {}, options = {}, parentEffectId = 0, name = 'anonymous', cont, ) { ... const task = newTask(parentEffectId, name, iterator, cont) const mainTask = { name, cancel: cancelMain, isRunning: true } const taskQueue = forkQueue(name, mainTask, end) ... next() return task function next(arg, isErr){ ... if (!result.done) { digestEffect(result.value, parentEffectId, '', next) } ... } }
其中 digestEffect 就执行了 effectTriggerd() 和 runEffect(),也就是执行 effect,其中 runEffect() 中定义了不同 effect 执行相对应的函数,每一个 effect 函数都在 proc.js 实现了。
除了一些核心方法之外,redux-saga 还提供了一系列的 helper 文件,这些文件的作用是返回一个类 iterator 的对象,便于后续的遍历和执行, 在此不具体分析。
上面是我整理给大家的,希望今后会对大家有帮助。
相关文章:
以上がredux-sagaの使い方、redux-sagaを使うための方法やテクニックとは?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。