react-redux: async promise
1、the simple sample
action: 事实上,只是返回个一个至少包含type的对象{ },用于reducer接收。
import {RECEIVE_DATA} from "constant/ActionType";
import MessageService from "service/demo-service/MessageService";
const _messageService = new MessageService();
function receiveData(data) {
return {
type: RECEIVE_DATA,
list: data
}
}
export function fetchData() {
return (dispatch) => {
_messageService.getMessage().then(res => {
dispatch(receiveData(res.data))
})
}
}
reducer:对应某个actionType有相应的数据返回
import {RECEIVE_DATA} from "constant/ActionType";
const initialState = {
message: []
};
const MessageReducer = function (state = initialState, action) {
switch (action.type) {
case RECEIVE_DATA:
return {message: action.list};
default:
return state;
}
};
export default MessageReducer;
use:
const mapStateToProps = state => {
return {
messageList: state.MessageReducer.message
}
};
const mapDispatchToProps = {
fetchMessageList: fetchData
};
componentDidMount() {
this.props.fetchMessageList();
}
2、Async Action Helpers
// status constants
export const REQUEST = 'request';
export const SUCCESS = 'success';
export const FAILURE = 'failure'; export const createAsyncAction = (type, promiseCreator) => {
return () => (dispatch, getState) => { dispatch({ type, readyState: REQUEST }); const promise = promiseCreator();
return promise.then(
data => {
dispatch({ type, readyState: SUCCESS, data });
},
error => {
dispatch({ type, readyState: FAILURE, error });
}
);
};
};
use:
function latestPostsRequest() {
return fetch(`https://www.reddit.com/latest.json`)
.then(
response => response.json(),
error => console.log('An error occured.', error)
)
}
const FETCH_LATEST_POSTS = 'Actions/Posts/FetchLatest';
export const fetchLatestPosts = createAsyncAction(FETCH_LATEST_POSTS, latestPostsRequest);
result:
import { fetchLatestPosts } from '../actions/posts';
// in some component...
dispatch(fetchLatestPosts);
// immediately:
// { type: 'Actions/Posts/FetchLatest', readyState: 'request' }
// after success:
// {
// type: 'Actions/Posts/FetchLatest',
// readyState: 'success',
// data: (some blob of json data from the api)
// }
3、More sophisticated actions: Args and Thunks
export const createAsyncAction = (type, promiseCreator) => {
// get hash of args to pass through to promise creator
return (args = {}) => (dispatch, getState) => {
dispatch({ args, type, readyState: REQUEST });
// pass args through
let promise = promiseCreator(args);
// handle thunk-style promises
if (typeof promise === 'function') {
promise = promise(dispatch, getState);
}
return promise.then(
data => {
dispatch({ args, type, readyState: SUCCESS, data });
},
error => {
dispatch({ args, type, readyState: FAILURE, error });
}
);
};
};
use:
import { createAsyncAction } from 'helpers/async';
import fetch from 'isomorphic-fetch'
// (pretend for a moment that such an action exists)
import { navigate, SIGN_IN } from 'actions/navigation';
// takes args: { subreddit }
function postsWithLikesRequest({ subreddit }) {
// returns a 'thunk promise' that uses dispatch and getState
return function(dispatch, getState) {
const userId = getState().userId;
if(!userId) {
// we have the dispatch function in our async action!
dispatch(navigate(SIGN_IN))
} else {
return fetch(`https://www.reddit.com/r/${subreddit}.json?likes=true&userId=${userId}`)
.then(
response => response.json(),
error => console.log('An error occured.', error)
)
}
}
}
const FETCH_POSTS_WITH_LIKES = 'Actions/Posts/FetchWithLikes';
export const fetchPostsWithLikes = createAsyncAction(FETCH_POSTS_WITH_LIKES, postsWithLikesRequest)
async reducer:
// helpers/async.js
import { combineReducers } from 'redux'; export const createAsyncReducer = (type) => {
const loading = (state = false, action) => {
if (action.type === type) {
return action.readyState === REQUEST;
}
return state;
}; const data = (state = null, action) => {
if (action.type === type) {
return action.data;
}
return state;
}; const error = (state = null, action) => {
if (action.type === type) {
return action.error;
}
return state;
}; return combineReducers({ loading, data, error });
};
use async reducer:
import { createAsyncReducer } from 'helpers/async';
import { FETCH_LATEST_POSTS } from 'actions/posts';
//
// Here's the reducer we can create with the helper
//
export const latestPosts = createAsyncReducer(FETCH_LATEST_POSTS);
//
// It's functionally equivalent to writing all this:
//
export function latestPosts(state = { loading: false, data: null, error: null }, action) {
if(action.type === FETCH_LATEST_POSTS) {
switch(action.readyState) {
case 'request':
return { loading: true, data: null, error: null };
case 'success':
return { loading: false, data: action.data, error: null };
case 'failure':
return { loading: false, data: null, error: action.error };
}
}
return state;
}
5、More configuration
The naive reducer returned by the simple createAsyncReducerfunction above is pretty useful for simple data fetches, but we often want to do more than just store the data from a request.
The actual createAsyncReducer function we use supports swapping in reducers that can respond to different actions or handle the request actions differently.
import {combineReducers} from "redux";
import {FAILURE, REQUEST, SUCCESS} from "constant/ActionType";
export const asyncReducerHelper = (type, reducers = {}) => {
let defaultReducers = {
loading: (state = false, action) => {
if (action.type === type) {
return action.readyState === REQUEST;
}
return state;
},
data: (state = null, action) => {
if (action.type === type && action.readyState === SUCCESS) {
return action.data;
}
return state;
},
error: (state = null, action) => {
if (action.type === type && action.readyState === FAILURE) {
return action.err;
}
return state;
},
};
return combineReducers(Object.assign({}, defaultReducers, reducers));
};
use:
import { createAsyncAction, createAsyncReducer } from 'helpers/async';
// I'm using the fetch api here, but you can substitute any http request libarary.
// For more about the fetch api: https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API
const postsRequest = (subreddit) => {
return fetch(`https://www.reddit.com/r/${subreddit}.json`)
.then(
response => response.json(),
error => console.log('An error occured.', error)
)
}
const FETCH_POSTS = 'Actions/Posts/Fetch';
// async action
export const fetchPosts = createAsyncAction(FETCH_POSTS, postsRequest);
// async reducer with shape { loading, data, error }
export const posts = createAsyncReducer(FETCH_POSTS);
react-redux: async promise的更多相关文章
- 在react+redux+axios项目中使用async/await
Async/Await Async/Await是尚未正式公布的ES7标准新特性.简而言之,就是让你以同步方法的思维编写异步代码.对于前端,异步任务代码的编写经历了 callback 到现在流行的 Pr ...
- react+redux教程(五)异步、单一state树结构、componentWillReceiveProps
今天,我们要讲解的是异步.单一state树结构.componentWillReceiveProps这三个知识点. 例子 这个例子是官方的例子,主要是从Reddit中请求新闻列表来显示,可以切换reac ...
- react+redux教程(六)redux服务端渲染流程
今天,我们要讲解的是react+redux服务端渲染.个人认为,react击败angular的真正“杀手锏”就是服务端渲染.我们为什么要实现服务端渲染,主要是为了SEO. 例子 例子仍然是官方的计数器 ...
- react+redux教程(一)connect、applyMiddleware、thunk、webpackHotMiddleware
今天,我们通过解读官方示例代码(counter)的方式来学习react+redux. 例子 这个例子是官方的例子,计数器程序.前两个按钮是加减,第三个是如果当前数字是奇数则加一,第四个按钮是异步加一( ...
- 实例讲解基于 React+Redux 的前端开发流程
原文地址:https://segmentfault.com/a/1190000005356568 前言:在当下的前端界,react 和 redux 发展得如火如荼,react 在 github 的 s ...
- 一个 React & Redux的目录树
|-----------------------------------------| | | | React & Redux | | | |------------------------- ...
- 使用react+redux+react-redux+react-router+axios+scss技术栈从0到1开发一个applist应用
先看效果图 github地址 github仓库 在线访问 初始化项目 #创建项目 create-react-app applist #如果没有安装create-react-app的话,先安装 npm ...
- webpack+react+redux+es6开发模式
一.预备知识 node, npm, react, redux, es6, webpack 二.学习资源 ECMAScript 6入门 React和Redux的连接react-redux Redux 入 ...
- react+redux官方实例TODO从最简单的入门(6)-- 完结
通过实现了增-->删-->改-->查,对react结合redux的机制差不多已经了解,那么把剩下的功能一起完成吧 全选 1.声明状态,这个是全选状态 2.action约定 3.red ...
- react+redux官方实例TODO从最简单的入门(1)-- 前言
刚进公司的时候,一点react不会,有一个需求要改,重构页面!!!完全懵逼,一点不知道怎么办!然后就去官方文档,花了一周时间,就纯react实现了页面重构,总体来说,react还是比较简单的,由于当初 ...
随机推荐
- PAT 1095 Cars on Campus
1095 Cars on Campus (30 分) Zhejiang University has 8 campuses and a lot of gates. From each gate we ...
- beego——事务处理和命令模式
1.事务处理 ORM 可以简单的进行事务操作. o := NewOrm() err := o.Begin() // 事务处理过程 ... ... // 此过程中的所有使用 o Ormer 对象的查询都 ...
- POJ_2533 Longest Ordered Subsequence【DP】【最长上升子序列】
POJ_2533 Longest Ordered Subsequence[DP][最长递增子序列] Longest Ordered Subsequence Time Limit: 2000MS Mem ...
- Codeforces Round #396 (Div. 2) D. Mahmoud and a Dictionary
地址:http://codeforces.com/contest/766/problem/D 题目: D. Mahmoud and a Dictionary time limit per test 4 ...
- c#的yield return
4.1 迭代器块 一个迭代器块(iterator block)是一个能够产生有序的值序列的块.迭代器块和普通语句块的区别就是其中出现的一个或多个yield语句. yield return语句产生迭代的 ...
- python 课堂笔记-while
#Author:zyl age_of_oldboy = 56 count = 0 while count < 3: guess_age = int(input("guess age:& ...
- C语言中 ln(以自然对数e为底) lg(以十为底) 以及logab(以a为底,b为真数)的相关知识
总所周知,我们在高中学过对数函数,记作y=logax.下面是百度百科关于对数函数的描述: 对数的定义:一般地,如果ax=N(a>0,且a≠1),那么数x叫做以a为底N的对数,记作x=logaN, ...
- APPIUM API整理(python)---其他辅助类
App运行类 1.current_activity current_activity(self): 用法: print(driver.current_activity()) Retrieves the ...
- [Android]AndFix使用说明
AndFix使用说明 AndFix,全称是Android hot-fix.是阿里开源的一个热补丁框架,允许APP在不重新发布版本的情况下修复线上的bug.支持Android 2.3 到 6.0,并且支 ...
- spring security采用基于简单加密 token 的方法实现的remember me功能
记住我功能,相信大家在一些网站已经用过,一些安全要求不高的都可以使用这个功能,方便快捷. spring security针对该功能有两种实现方式,一种是简单的使用加密来保证基于 cookie 的 to ...