Immutable入门使用_在react项目中使用redux-immutable
定义
不可变数据 (Immutable Data
)就是一旦创建,就不能再被更改的数据。对 Immutable 对象的任何修改或添加删除操作都会返回一个新的 Immutable 对象。Immutable 实现的原理是持久化数据结构
( Persistent Data Structure),也就是使用旧数据创建新数据时,要保证旧数据同时可用且不变。同时为了避免 deepCopy 把所有节点都复制一遍带来的s性能损耗,Immutable 使用了 结构共享
(Structural Sharing),即如果对象树中一个节点发生变化,只修改这个节点和受它影响的父节点,其它节点则进行共享
,也就是Immutable会做按需改变,只会把需要更改的节点做改变,其它的节点没有变。
Immutable.js的优缺点
优点:
- 降低mutable带来的复杂度
- 节省内存
- 历史追溯性(时间旅行):时间旅行指的是,每时每刻的值都被保留了,想回退到哪一步只要简单的将数据取出就行,想一下如果现在页面有个撤销的操作,撤销前的数据被保留了,只需要取出就行,这个特性在redux或者flux中特别有用。
- 拥抱函数式编程:immutable本来就是函数式编程的概念,纯函数式编程的特点就是,只要输入一致,输出必然一致,相比于面向对象,这样开发组件和调试更方便。
缺点:
- 需要重新学习api。
- 资源包大小增加(源码5000行左右)。
- 容易与原生对象混淆:由于api与原生不同,混用的话容易出错。
使用Immutable.js
参考官网重点讲解数据不可变数据的创建、更新及比较方式。
安装
终端中执行yarn add immutable -S
,安装immutable到生产环境中。
List
- List 类似js中的数组
- 读取数据用
list.get(下标)
- 获取长度用
list.size
- 判断是否为List对象,
List.isList(list)
- 可以用js中数组的方法
push()
、unshift()
、concat()
、map()
import { List } from 'immutable'const list = List([1, 2, 3, 4])
const newList = list.push(5)
console.log(list, newList) //List{..} List{..}
console.log(list[4], newList[4]) // undefined undefined
console.log(list.get(4), newList.get(4)) // undefined 5
console.log(list.size, newList.size) // 4 5const arr = [1, 2, 3, 4]
console.log(List.isList(list), List.isList(arr)) // true false
控制台中打印结果如下:
List常用api
push()
在list尾部添加数据,返回新的immutable对象unshift()
在list首部添加数据,并返回新的immutable对象concat()
连接多个list,并返回新的immutable对象map()
对list 进行遍历,类似js数组中map的使用,返回一个新的immutable对象join()
将list数组拼接成字符串toJs()
将immutable对象转成js对象
import { List } from 'immutable'const list = List([1, 2, 3, 4])
const list1 = list.push(5)
const list2 = list1.unshift(0)
const list3 = list.concat(list1, list2)
const list4 = list.map(v => v * 2)console.log("list,",list.toJS()) //[1,2,3,4]
console.log("list1,",list1.toJS()) //[1,2,3,4,5]
console.log("list2,",list2.toJS()) //[0,1,2,3,4,5]
console.log("list3,",list3.toJS()) //[1,2,3,4,1,2,3,4,5,0,1,2,3,4,5]
console.log("list4,",list4.toJS()) //[2,4,6,8]
Map
- Map 数据结构和es6的map类似
- 更改值用
map.set(key,value)
,会返回一个新的变量 - 读取数据用
map.get(key)
- 使用
Map.isMap(obj)
,来判断是否是一个immutable.Map类型 - 将多个对象合并到一起使用
merge()
- 如果要转换成es6中的对象,使用
toObject()
或者toJS()
import { Map } from 'immutable'const map = Map({a: 1,b: 2,c: 3})// immutable数据每次都是生成新的再重新调用set进行修改,所以需要重新赋值给一个新的变量
const newMap = map.set('b', 20) console.log(map, newMap) // Map{...} Map{...}
console.log(map.b, newMap.b) // undefined undefined
console.log(map.get('b'), newMap.get('b')) // 2 20const obj = {a: 1,b: 2,c: 3}
console.log(Map.isMap(map), Map.isMap(obj)) // true false,
控制台中打印结果如下:
Map常用api
map()
对map数据进行遍历,第一个参数对应map中的value,第二个参数对应map中的keytoJs()
将immutable对象转成js对象toObject()
将immutable对象转成js对象fromJs()
将js对象转换成immutable对象merge()
将immutable对象或者js对象进行合并,当集合发生冲突时,后面的数据会覆盖前面的数据,它只能用在immutable对象上
import { Map } from 'immutable'const alpha = Map({a: 1,b: 2,c: 3})
const objKeys = alpha.map((value, key) => key)
console.log(objKeys.join()) // a, b, cconst map1 = Map({a: 1, b: 2})
const map2 = Map({c: 3,d: 4})
const obj1 = {d: 400,e: 50}
const obj2 = {g: 500,h: 60}const mergedMap = map1.merge(map2, obj1)
const mergeObj1=map2.merge(obj1) //obj1里的d会覆盖map2里面的d
// const mergeObj2=obj1.merge(obj2) //报错 merge()是immutable对象的方法console.log(mergedMap.toObject()) //{a: 1, b: 2, c: 3, d: 400, e: 50}
console.log(mergedMap.toJS()) //{a: 1, b: 2, c: 3, d: 400, e: 50}
console.log(mergeObj1.toJS()) //{c: 3, d: 400, e: 50}
equals & is
- equals is 用来比较两个
immutable对象
是否相等
map.equals(anotherMap)
如果值相等的话,返回trueis(map,anotherMap)
如果值相等的话,返回true
import { Map, is } from 'immutable'const map = Map({a: 1,b: 2,c: 3})const anotherMap = Map({a: 1,b: 2,c: 3})console.log(map == anotherMap) // false
console.log(map === anotherMap) // false
console.log(map.equals(anotherMap)) // 使用equals进行比较 true
console.log(is(map, anotherMap)) // 使用is进行比较 true
嵌套数据结构
mergeDeep()
,像merge()一样用来合并数据,可以对嵌套数据进行深度递归getIn()
用来获取嵌套数据的值,nested.getIn([‘a’,‘b’,‘c’]) 表示获取a.b.c的值updateIn()
用来更新嵌套数据的值,第一个参数表示要更新的数据,第二个参数表示更新后的值
const { fromJS } = require('immutable');
const obj={a:{b:{c:[3,4,5]}}
}
const nested = fromJS(obj);const nested2 = nested.mergeDeep({ a: { b: { d: 6 } } });
// Map { a: Map { b: Map { c: List [ 3, 4, 5 ], d: 6 } } }console.log(nested2.getIn([ 'a', 'b', 'd' ])); // 6const nested3 = nested2.updateIn([ 'a', 'b', 'd' ], value => value + 1);
console.log(nested3);
// Map { a: Map { b: Map { c: List [ 3, 4, 5 ], d: 7 } } }const nested4 = nested3.updateIn([ 'a', 'b', 'c' ], list => list.push(6));
console.log(nested4)
// Map { a: Map { b: Map { c: List [ 3, 4, 5, 6 ], d: 7 } } }
在redux中使用Immutable
redux官网推荐使用redux-immutable进行redux和immutable的集成。几个注意点:
安装
终端中执行yarn add redux-immutable -S
,安装redux-immutable到生产环境中。
使用
redux
中,利用combineReducers
来合并多个reduce
, redux
自带的combineReducers
只支持原生js形式的,所以需要使用redux-immutable
提供的combineReducers
来代替。
下面用一个计数器的案例来分析:
// 使用redux-immutable提供的combineReducers方法替换redux里的combineReducers
import {combineReducers} from 'redux-immutable'
import counter from './reducerCounter'
import anotherReducer from './anotherReducer'const rootReducer = combineReducers({counter,anotherReducer
});export default rootReducer;
reducer
中的initialState
也需要初始化成immutable
类型,使用set
或setIn
来更改值, get
或者getIn
来取值,比如一个counter的reducer:
import { Map } from 'immutable'
import ActionTypes from '../actions'const initialState = Map({count: 0})export default (state = initialState, action) => {switch (action.type) {case ActionTypes.INCREAMENT:return state.set('count', state.get('count') + 1) case ActionTypes.DECREAMENT:return state.set('count', state.get('count') - 1)default:return state}
}
state
成为了immutable
类型,connect
的mapStateToProp
也需要相应的改变,永远不要在mapStateToProps里使用toJS
方法,因为它永远返回一个新的对象。
const mapStateToProps = state => ({count: state.getIn(['counter', 'count'])
})
在shouldComponentUpdate
里就可以使用immutable.is
或者instance.equals
来进行数据的对比了。
Immutable入门使用_在react项目中使用redux-immutable相关推荐
- 优雅的在React项目中使用Redux
概念 首先我们会用到哪些框架和工具呢? React UI框架 Redux 状态管理工具,与React没有任何关系,其他UI框架也可以使用Redux react-redux React插件,作用:方便在 ...
- 如何优雅地在React项目中使用Redux
前言 或许你当前的项目还没有到应用Redux的程度,但提前了解一下也没有坏处,本文不会安利大家使用Redux 概念 首先我们会用到哪些框架和工具呢? React UI框架 Redux 状态管理工具,与 ...
- 优雅的在React项目中使用Redux 1
或许你当前的项目还没有到应用Redux的程度,但提前了解一下也没有坏处 首先我们会用到哪些框架和工具呢? React UI框架 Redux 状态管理工具,与React没有任何关系,其他UI框架也可以使 ...
- antd 实现pdf 预览_解决react项目中PDF的显示与打印问题
最近项目中有这样一个需求: 1. 页面中可以显示pdf 2. 不希望把整个页面打印下来,只打印显示PDF的部分,可以使用浏览器自带打印功能 PDF文件的显示 拿到这个需求,真时一头雾水.因为没有做过类 ...
- react本地储存_如何在React项目中利用本地存储
react本地储存 以及为什么要这么做. 本地存储是现代Web浏览器固有的Web API. 它允许网站/应用程序在浏览器中存储数据(简单和有限),从而使数据在以后的浏览器会话中可用. 在开始学习本教程 ...
- 中使用js修改变量值_谈一谈css-in-js在React项目中的使用
一.什么是css-in-js 参考:[css in js 简介] 简单来说,传统的前端方案推崇"关注点分离"原则,HTML.CSS.JavaScript 应该各司其职,进行分离. ...
- react 使用 leaflet 百度地图_【React】react项目中应用百度地图添加起始点绘制路线...
如图:项目中百度地图的应用添加起始点.终点并绘制路线 在展示代码的时候首先展示一下后台返回给我的接口 { 其中position_list参数代表的是用户的行驶点, area参数代表的是服务区的坐标点, ...
- react前端显示图片_如何在react项目中引用图片?
如何在react项目中引用图片?本文码云笔记将为大家整理在react项目中插入图片以及背景图片的方法,希望对需要的小伙伴提供一些参考. 在react项目中插入图片以及背景图片的方法共有2种: 1.im ...
- React 项目中使用Echarts
直接上代码吧: react 项目中添加Echarts 相关模块 npm install echarts --save 代码: import React from 'react' import * as ...
最新文章
- linux文件IO与内存映射:用户空间的IO缓冲区
- Ubuntu 想要更新源 报错 “E: 无法获得锁 /var/lib/dpkg/lock-frontend - open (11: 资源暂时不可用)”
- Linux-什么是二进制包,源码包,RPM包,软件仓库
- CreateEvent(NULL,TRUE,FALSE,NULL);的解释
- 【今晚9点】:对话黄琦——从FB到快手,短视频领域里的“实习生”
- 使用QT的一些小Tipster
- android 八核手机,八核手机
- 已知a类被打包在packagea_2021考研干货:199管理类联考综合逻辑归纳习题(1)
- normalize()方法
- 高并发情况下修改系统参数
- 使用IntelliJ IDEA配置Erlang开发环境
- 拓端tecdat|R语言 PCA(主成分分析),CA(对应分析)夫妻职业差异和马赛克图可视化
- 游戏一般用什么编程语言开发?
- windows反馈中心服务器,Windows 管理中心入门
- 知识图谱(KG)中的同义词挖掘
- C语言sqrt求平方根函数注意点
- 删除脚注或尾注的横线
- u盘上传百度网盘照片显示服务器错误,百度云上传文件失败怎么办-百度云上传文件失败的解决方法 - 河东软件园...
- 用python画小猪佩奇的编码_如何用Python代码画小猪佩奇
- 浅谈智能计算机的未来发展,浅谈计算机人工智能科学技术应用及发展
热门文章
- TP-LINK交换机安全配置的设置方法
- 我和计算机比本领教案反思,《比本领》教学反思范文
- 皆大欢喜!iPhone不再耗电,续航增加就靠它
- 用大津法进行omr识别
- c# mysql 汉字乱码_C#连接MySQL汉字乱码
- 第六次作业--结对编程
- 【Android自定义View实战】之仿QQ运动步数圆弧及动画,Dylan计步中的控件StepArcView
- Android开发给apk签名
- 如何查看Tomcat版本号
- the import java.sql cannot be resolved_the import XXXX cannot be resolved 解决方法