在学react的是,发现一旦我们封装好了我们的组件,那么我们的项目就跟搭积木一样简单快速,可是我们发现了一个问题,在一个页面往往会嵌套很多的组件,子组件必须要通过父组件传递参数才能渲染出数据,我们回想一下我们之前构建过的所有react应用,数据都是由最顶层父组件(页面组件)一层层向下传递的。

这也是深层次的组件之间通讯困难的原因:数据的传递是单向的,子组件的数据只能就近获取,但是真正的数据源却离得太远,没有捷径可以直接通知数据源更新状态。

redux的出现改变了react的这种窘迫处境,它提供了整个应用的唯一数据源store,这个数据源是随处可以访问的,不需要靠父子相传,并且还提供了(间接)更新这个数据源的方法,并且是随处可使用的!

那么redux是如何工作的呢?这里我借用阮一峰react讲解redux的图来说明:

用户看到的是能是我们的页面,那么用户如何跟我们的数据互动呢。首先初始化页面,我们会渲染store里面的初始数据,
store 是redux提供的唯一数据源,它存储了整个应用的state,并且提供了获取state的方法,即store.getState()
现在用户点击了某个按钮,那么这个时候用户就触发了一个动作,称之为action,action只是记录了你刚才"做了什么动作",但是就是知道你干了这个事情,至于事情会发生什么作用,这个并不是你说的算的,就要引入我们的reducer了,如果说action就是你平常做的事,那么reducer就是国家的法律,根据你的事法律会给你相应的后果。
 
1.我们先看action
action是一个对象,记录本次事件的类型,以便对应相应的结果,除了type属性的其他属性我们称之为载荷(payload),除了type属性外,其他属性都是非必选的。
{
type: 'CHANGE_LIST',
data: [1,2,3,4]//这个是假数据,这里当做需要变成的数据,一般是后台拿到的
}

接下来我们要触发这个事件

2.在redux中触发这个事件的方式就是:

dispatch({
type: 'CHANGE_LIST',
data:[1,2,3,4]
})

3.到这里我们触发了事件,那么事件会产生啥效果呢,这里就提到我们的reducer

假设我们现在有一个列表组件,那么它的数据格式应该可以如下:

// store.getState()

{
list: {
products:['name','age','haha','gege']
} }

现在我们做了步骤2的action操作,那么reducer应该是这样的:

function reducer(state, action) {
switch (action.type) {
case 'CHANGE_LIST':
return {
list: {
products:action.data//修改给定的数据
}
}
default:
return state // 没有匹配的action type,返回原来的state
}
}
 
 到此为止我们知道这个操作会产生的效果就是原来的数组products从   ['name','age','haha','gege']  变成了  [1,2,3,4]。
4.拆分reducer
我们在react真实的项目肯定包含许多的组件,不能把所有组件的reducer写在一个函数里面,这样会显得更复杂以及文件特别大,所以我们把上面的reducer改写一下
function listReducer(state, action) {
switch (action.type) {
case 'CHANGE_LIST':
return {
products:action.data//修改给定的数据
}
default:
return state // 没有匹配的action type,返回原来的state
}
} function reducer(state, action) {
return {
list: listReducer(state.list, action),
}
}

可以看出达到的效果是一致的,刚好redux为reducer的合并提供了一个简单的方法combineReducers,

function list(state, action) {
switch (action.type) {
case 'CHANGE_LIST':
return {
products:action.data
} default:
return state // 没有匹配的action type,返回原来的state
}
} function dialog(state, action) {
switch (action.type) {
case 'SHOW_DIALOG':
return {
status: true
} case 'CLOSE_DIALOG':
return {
status: false
} default:
return state // 没有匹配的action type,返回原来的state
}
} export default combineReducers({
list,
dialog
})

5.生成store,在我们项目的入口文件app.js中:

import React, { Component, PropTypes } from 'react'
import ReactDom from 'react-dom' // 引入redux
import { createStore, applyMiddleware } from 'redux'
import { Provider, connect } from 'react-redux' // 引入reducer
import * as reducers from './redux/reducer.js'
import { combineReducers } from 'redux' //引入组件
import Index from './component/index.js' //把多个组件的reducer合成总的reducer
const reducer = combineReducers(reducers) // 创建store,这个是整个应用唯一的
const store = createStore(reducer) ReactDom.render(
<Provider store={store}>
<Index />
</Provider>,
document.getElementById('root')
)

然后看看我们的Index组件是个啥样子:

import React, { Component } from 'react'
import { connect } from 'react-redux' import List from './list.js' class Index extends Component {
constructor(props) {
super(props);
}
render () {
return (
<div>
<List />
</div>
)
}
} function mapStateToProps(state) {
return state //对应本组件需要的传入的props
} export default connect(mapStateToProps)(Index)//关联到store上

下面最重要的看看List组件:

import React, { Component } from 'react'
import PropTypes from 'prop-types'
import { connect } from 'react-redux' class List extends Component {
constructor(props) {
super(props);
}
render() {
var html = this.props.products.map(function(val,i) {
return <li key={i}>{val}</li>
}) return (
<div>
<ul>
{html}
</ul>
<button onClick={this.props.change}>改变数组</button>
</div>
)
}
} function mapStateToProps(state) {
var info = state.list return {
products:info.products//对应本组件props需要的属性products
}
} function mapDispatchToProps(dispatch) {
return {
change () {//这里change方法对应的是这个组件需要外部传入的change方法
dispatch({
type: 'CHANGE_LIST',
data:[1,2,3,4]
})
}
}) }
}
} export default connect(mapStateToProps,mapDispatchToProps)(List)

这里只是简单介绍了redux基本知识,但是在实际的项目中如何应用呢,实际的项目react和redux结合如何搭建框架呢,这个下篇文章将会列出一个真实的项目。

react+redux基础用法的更多相关文章

  1. Flux --> Redux --> Redux React 入门 基础实例使用

    本文的目的很简单,介绍Redux相关概念用法 及其在React项目中的基本使用 假设你会一些ES6.会一些React.有看过Redux相关的文章,这篇入门小文应该能帮助你理一下相关的知识 一般来说,推 ...

  2. 【温故知新】—— React/Redux/React-router4基础知识&独立团Demo

    前言:React专注View层,一切皆组件:全部使用ES6语法,最新版本为React16. Redux是专注于状态管理的库,和react解耦:单一状态,单向数据流.[独立团github地址] 一.Re ...

  3. react基础用法二(组件渲染)

    react基础用法二(组件渲染) 如图所示组件可以是函数 格式:function 方法名(){ return <标签>内容</标签>} 渲染格式: <方法名 />  ...

  4. react基础用法一(在标签中渲染元素)

    react基础用法一(渲染元素) 如图所示最简单的变量使用方法 格式 let 变量名称 = 赋值: 渲染格式直接用 {变量名称} 就可以直接渲染到页面 如图所示第二种渲染方法 格式 const 变量名 ...

  5. react+redux教程(四)undo、devtools、router

    上节课,我们介绍了一些es6的新语法:react+redux教程(三)reduce().filter().map().some().every()....展开属性 今天我们通过解读redux-undo ...

  6. 实例讲解基于 React+Redux 的前端开发流程

    原文地址:https://segmentfault.com/a/1190000005356568 前言:在当下的前端界,react 和 redux 发展得如火如荼,react 在 github 的 s ...

  7. Immutable.js 以及在 react+redux 项目中的实践

    来自一位美团大牛的分享,相信可以帮助到你. 原文链接:https://juejin.im/post/5948985ea0bb9f006bed7472?utm_source=tuicool&ut ...

  8. react + redux 实现幻灯片

    写在前面: 这一篇是我 使用scss + react + webpack + es6实现幻灯片 的进阶篇,效果请点我,将会使用上redux的基础用法,因为一开始没有理解好redux的用法,单纯看文档, ...

  9. react+redux教程(三)reduce()、filter()、map()、some()、every()、...展开属性

    reduce().filter().map().some().every()....展开属性   这些概念属于es5.es6中的语法,跟react+redux并没有什么联系,我们直接在https:// ...

随机推荐

  1. Qt Qwdget 汽车仪表知识点拆解3 进度条编写

    先贴上效果图,注意,没有写逻辑,都是乱动的 这篇我来说说左侧的这个进度条的实现原理,其实更简单,哈哈哈 有一个大的widget,根据素材,我放了10个label 剩下的就是写一个函数,根据数据的不同, ...

  2. 用Python 的一些用法与 JS 进行类比,看有什么相似?

    Python 是一门运用很广泛的语言,自动化脚本.爬虫,甚至在深度学习领域也都有 Python 的身影.作为一名前端开发者,也了解 ES6 中的很多特性借鉴自 Python (比如默认参数.解构赋值. ...

  3. VSCode 前端必备插件

    VSCode 前端必备插件 Debugger for Chrome 让 vscode 映射 chrome 的 debug功能,静态页面都可以用 vscode 来打断点调试 { "versio ...

  4. 1072 Gas Station (30 分)(最短路径)

    #include<bits/stdc++.h> using namespace std; ; int n,m,k,Ds; int mp[N][N]; int dis[N]; int vis ...

  5. Elasticsearch 监控和部署

    Elasticsearch: ! [ https://elasticsearch.cn/book/elasticsearch_definitive_guide_2.x/_cluster_health. ...

  6. neutron DVR

    DVR 简介 DVR 提出的背景 在 Neutron 的网络环境中,跨子网的虚机通信是需要通过 Neutron 的路由器.这既包括不同子网的虚拟机之间的通信,又包括虚拟机与外网之间的通信.在 DVR ...

  7. vue 介绍的拓展

    计算属性 <div id="example"> <p>Original message: "{{ message }}"</p&g ...

  8. clone项目到本地

    clone项目到本地 1.然后在本地建立接受代码的文件夹,然后cd 到这个目录 (克隆版本库的时候,所使用的远程主机自动被git命名为origin.如果想用其他的主机名,需要用git clone命令的 ...

  9. Delphi xe7组件和控件的安装方法

    暂时我所遇到的所有控件安装方法大体与下面两种相同. 若有不同大家提出来,一起想办法解决. .dproj格式的组件安装方法: raise组件 安装详细步骤如下: 一.设置搜索路径1. 将本包中的文件连同 ...

  10. MySQL触发器写法

    触发器创建语法四要素:1.监视地点(table) 2.监视事件(insert/update/delete) 3.触发时间(after/before) 4.触发事件(insert/update/dele ...