前言

  前几天,我在博文【前端】一步一步使用webpack+react+scss脚手架重构项目 中搭建了一个react开发环境。然而在实际的开发过程中,或者是在对源码的理解中,感受到react中用的最多的,就是redux了,于是打开文档学习了一番。在这里做一些记录。

redux概念

  redux是什么?它是用来管理状态的。在react开发中,我们经常会遇到一种情况,组件与父组件之间的通信,组件与组件之间的通信,

其中组件与父组件的通信通过props来完成。

  

/***********parent****************/
<Number
value={this.state.number}
/> /***********number****************/
<div >{this.props.value}</div>

number内部使用props获取父组件自己的state,通过setState更改父组件的state.number可以实现number组件的实时更新。

那么组件与组件之间的通信呢?例如我们有a,b两个组件,现在a要给b转发一个激活状态active,一般思路是,在同一个父组件下,使用props传递回调函数的方式。

/*********组件 a**********/
class A{
.....
changeActive(){
this.props.changebActive();
}
render(){
return <button onClick={this.changeActive}></button>
}
}
/*********组件 b**********/
class B{
render(){
return <div>状态:{this.props.active}</div>
}
}
/*********组件 parent**********/
class parent{
......
changebActive(){
this.setState({
active:"开启"
})
}
render(){
return <div>
<A
changebActive={this.changebActive.bind(this)}
/>
<B
active={this.state.active}
/>
</div>
}
}

通过A点击调用到parent的 changebActive方法,设置state更新b,这种方式非常容易使项目变得不可预测,并且state管理复杂,没错,redux很大程度就是用来解决这个问题的,它在react中的思想是:

将组件状态统一放到同一个超级父组件,再由这个超级父组件用更专业的方法分发props给所有的子组件,无论嵌套多少层。由超级父组件统一管理应用中所有的state。

---这就是redux。无论api名词如何晦涩难懂,它的核心思想是在开发中浮现的。

redux的API

  一、所有的状态,保存在一个对象里面。

    所有的状态都保存在一个对象里面,redux基本三大原则之一,单一数据源,一个应用所有的状态被保存在一个对象里,这个对象通过store管理,想象一下,整个页面的状态被存储在一个对象里面,其实类似一个配置文件,你可以从服务器读取,可以从任何地方改变。view则是同步的,这看起来更精炼不是么。

  二、store

  Store 就是保存状态数据对象的地方,你可以把它看成一个容器。整个应用只能有一个 Store。

Redux 提供createStore这个函数,用来生成 Store。createStore传入了一个函数这个函数将接收到两个参数,一个state,一个action,action就是由子组件发出的更改请求,state就是那个单一的state对象。

import { createStore } from 'redux';
const store = createStore(fn);

  三、action

发起状态改变的起点是action,由子组件发出action,子组件通过props调用回调函数,通知父组件发出action。

触发:

store.dispatch(action);

  

  四、store.dispatch

  这个方法就是发出action的方法,一般传入action,调用这个方法之后,会触发在createStore中传入的函数。

  五、store.getState

   它用来获取当前的state对象

  六、reducer

这个就是createStore中传入的函数的名词解释,这个函数的执行必须返回一个新的state:

export default (state={number:0},action)=>{//每次调用都会传入state和当前通知的action
switch(action.type){
case "NUMBER_ADD": //发过来的action如果是NUMBER_ADD 就返回number++
state.number++;
break;
case "NUMBER_LESS":
state.number--;
break;
}
return Object.assign({},state); //必须返回一个新的state ,可以使用assign合并
}

注意这里用了es6,给state添加了一个默认值,{number:0}。

  六、store.subscribe()

Store 允许使用store.subscribe方法设置监听函数,一旦 State 发生变化,就自动执行这个函数。

import { createStore } from 'redux';
const store = createStore(reducer); store.subscribe(render);

比如上面我在生成store传入reducer函数之后,使用store.subscribe()设置监听函数为render,则每次state改变了,就会执行render函数,刷新视图。

概念名词太多,不如实践

  实践一个number输入框的加减组件,你就能大概知道redux究竟是怎么工作的。

  一、准备

    首先你需要搭建一个react的开发环境,需要一个index.html 输出目录等。详见我的另一篇博文:【前端】一步一步使用webpack+react+scss脚手架重构项目  。

然后你需要一定的react开发实践,React有props和state两个属性: props意味着父组件分发下来的属性,state意味着组件内部可以自行管理的状态,并且整个React没有数据向上回溯的能力,也就是说数据只能单向向下分发,或者自行内部消化。

理解这个是理解React和Redux的前提。

  你需要npm安装redux:

    num install redux react-redux --save-dev

  二、代码

  1.number.js

   

import React,{Component} from "react";

const propTypes = {
add:React.PropTypes.func.isRequired,
less:React.PropTypes.func.isRequired,
value:React.PropTypes.object.isRequired
}
class Number extends Component{
add(){
this.props.add({type:"NUMBER_ADD"}) //调用app.js中传过来的箭头函数,传入了type为NUMBER_ADD的action
}
less(){
this.props.less({type:"NUMBER_LESS"})
}
render(){
return <div>
<input type="text" value={this.props.value.number}/>
<button onClick={this.add.bind(this)}>
+
</button>
<button onClick={this.less.bind(this)}>
-
</button>
</div>
}
}
Number.propTypes = propTypes;
export {Number};

注意那两个click回调,发出了action,以及那个this.props.value.number的引用,props传入了state,获取了state.number

2.app.js

import React from "react";
import ReactDOM from "react-dom"; import {createStore} from "redux"; //导入createStore import {Number} from "./number.js";
import reducers from "./reducers.js";
const store = createStore(reducers); //生成store const content = document.querySelector(".content");
const render = ()=> ReactDOM.render(
<div>
<Number
value={store.getState()}
add={(action)=>store.dispatch(action)} //传入一个函数,传入发送过来的action,由reducers处理之后返回state,
less={(action)=>store.dispatch(action)}
/>
</div>,
content
);
render();
store.subscribe(render); //在reducers处理之后,返回了state,然后触发了render,更新视图

3.reducers.js

export default (state={number:0},action)=>{//每次调用都会传入state和当前通知的action
switch(action.type){
case "NUMBER_ADD": //发过来的action如果是NUMBER_ADD 就返回number++
state.number++;
break;
case "NUMBER_LESS":
state.number--;
break;
}
return Object.assign({},state); //必须返回一个新的state ,可以使用assign合并
}

这就是createStore传入的函数,当发出action,store会自动调用它传入state,action 返回了一个新的state.

Redux 的基本用法就介绍到这里,下一篇介绍它的高级用法:中间件和异步操作。为什么是下一篇?因为我也还没学。

资源

  阮一峰的教程

  number——demo的github

  知乎上大神对redux的易懂解读

--------------------------------

转载必须在页头注明出处,此前多文章被转载不署名,自重

【前端】react and redux教程学习实践,浅显易懂的实践学习方法。的更多相关文章

  1. 【前端,干货】react and redux教程学习实践(二)。

    前言 这篇博文接 [前端]react and redux教程学习实践,浅显易懂的实践学习方法. ,上一篇简略的做了一个redux的初级demo,今天深入的学习了一些新的.有用的,可以在生产项目中使用的 ...

  2. React 与 Redux 在生产环境中的实践总结

    React 与 Redux 在生产环境中的实践总结 前段时间使用 React 与 Redux 重构了我们360netlab 的 开放数据平台.现将其中一些技术实践经验总结如下: Universal 渲 ...

  3. MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example 1.4 Labeling the Map

    MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example 1.4 Labeling the Map 一.前言 MapServer拥有非常灵活的标签 ...

  4. MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.7 Adding a wms layer

    MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.7 Adding a wms layer 前言 Add OGC WMS Layers( ...

  5. MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.6 Defining Projections and Extents

    MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.6 Defining Projections and Extents 一.前言 当在m ...

  6. MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.5 Adding a raster layer

    MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.5 Adding a  raster layer 一.前言 MapServer不仅支持 ...

  7. MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.3 Displaying Classes in a Layer

    MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.3 Displaying Classes in a Layer 一.前言 关于第一节的 ...

  8. MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.2 Static Map with Two Layers

    MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.2 Static Map with Two Layers 一.前言 上一篇博客< ...

  9. MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.1 A map with single layer

    MapServer Tutorial——MapServer7.2.1教程学习——第一节用例实践:Example1.1 A map with single layer 一.前言 开始MapServer用 ...

随机推荐

  1. HTML5浏览器定位navigator.geolocation.getCurrentPosition

    <!DOCTYPE html> <html> <body> <p id="demo">点击这个按钮,获得您的坐标:</p> ...

  2. Babel 入门指南

    Babel 入门指南 ​⚠️ 注意: Babel 可以与很多构建工具(如 Browserify.Grunt.Gulp 等)进行集成.由于本教程选择 Webpack ,所以只讲解与 Webpack 的集 ...

  3. 关于cas-client单点登录客户端拦截请求和忽略/排除不需要拦截的请求URL的问题(不需要修改任何代码,只需要一个配置)

    前言:今天在网上无意间看到cas单点登录排除请求的问题,发现很多人在讨论如何通过改写AuthenticationFilter类来实现忽略/排除请求URL的功能:突发奇想搜了一下,还真蛮多人都是这么干的 ...

  4. arcgis api for js入门开发系列十一地图统计图

    上一篇实现了demo的叠加SHP图层,本篇新增地图统计图,截图如下: 地图统计图实现的思路如下:利用拓展arcgis api的js文件(MapChartGraphic.js以及MapChartGrap ...

  5. FutureTask分析(1.8)

    FutureTask简介 FutureTask用于异步计算,也就是支持异步执行并返回结果.FutureTask本身是一个Runable,所以可以交给Thread来运行,在提交给Thread运行后,可以 ...

  6. linux 下tomcat的安装

    写在前面: 由于项目使用jdk1.6开发,所以对应服务器应安装jdk1.6和tomcat6 --- 1.环境变量的配置: 打开/etc/bashrc配置环境变量 JAVA_HOME=/usr/apps ...

  7. Flash TextField selectable bug block TextEvent.Link solution

    There is an old version Felx SDK bug(in my case it's Flex SDK v3.3.0.4852) that when TextField.selec ...

  8. thinkphp3.2自定义常量

    在项目文件夹 (如:Home) 中的Common文件夹下新建function.php //thinkphp3.2.2版本加入如下语句: define('XXX', XXX); //第一个参数是常量名, ...

  9. Http批量异步发送和数据保存

    先说需求. 有个服务程序定时扫描指定文件夹下一个所有文件,文件包含了多个用户(客户)信息及对应的http发送地址和发送数据.现在该服务程序需要提取这些用户信息,然后批量进行发送:发送完后需要将http ...

  10. JavaScript对象属性访问的两种方式

    JavaScript对象属性访问的两种方式 object.attribute object["attribute"] 例如: var employees = [ { "f ...