本文主要整理了Immutable.js经常使用API的使用。git
关于Immutable的定义,官方文档是这样说的:github
Immutable data encourages pure functions (data-in, data-out) and lends itself to much simpler application development and enabling techniques from functional programming such as lazy evaluation.算法
简而言之,Immutable数据就是一旦建立,就不能更改的数据。每当对Immutable对象进行修改的时候,就会返回一个新的Immutable对象,以此来保证数据的不可变。对于Immutable的好处及介绍,你们能够参考Immutable 详解及 React 中实践,这篇文章介绍的很清楚。数组
由于Immutable的官方文档有点晦涩难懂,本文只是用来整理Immutable经常使用的API的使用,便于使用与查询,想了解更详细的内容,请戳这里~数据结构
有不妥的地方欢迎你们指出~app
List
: 有序索引集,相似JavaScript中的Array。Map
: 无序索引集,相似JavaScript中的Object。OrderedMap
: 有序的Map
,根据数据的set()进行排序。Set
: 没有重复值的集合。OrderedSet
: 有序的Set
,根据数据的add进行排序。Stack
: 有序集合,支持使用unshift()和shift()添加和删除。Range()
: 返回一个Seq.Indexed类型的集合,这个方法有三个参数,start表示开始值,默认值为0,end表示结束值,默认为无穷大,step表明每次增大的数值,默认为1.若是start = end,则返回空集合。Repeat()
: 返回一个vSeq.Indexe类型的集合,这个方法有两个参数,value表明须要重复的值,times表明要重复的次数,默认为无穷大。Record
: 一个用于生成Record实例的类。相似于JavaScript的Object,可是只接收特定字符串为key,具备默认值。Seq
: 序列,可是可能不能由具体的数据结构支持。Collection
: 是构建全部数据结构的基类,不能够直接构建。用的最多就是List和Map,因此在这里主要介绍这两种数据类型的API。性能
做用:将一个js数据转换为Immutable类型的数据。ui
用法:fromJS(value, converter)
this
简介:value是要转变的数据,converter是要作的操做。第二个参数可不填,默认状况会将数组准换为List类型,将对象转换为Map类型,其他不作操做。lua
代码实现:
const obj = Immutable.fromJS({a:'123',b:'234'},function (key, value, path) { console.log(key, value, path) return isIndexed(value) ? value.toList() : value.toOrderedMap()) })
做用:将一个Immutable数据转换为JS类型的数据。
用法:value.toJS()
做用:对两个对象进行比较。
用法:is(map1,map2)
简介:和js中对象的比较不一样,在js中比较两个对象比较的是地址,可是在Immutable中比较的是这个对象hashCode
和valueOf
,只要两个对象的hashCode
相等,值就是相同的,避免了深度遍历,提升了性能。
代码实现:
import { Map, is } from 'immutable' const map1 = Map({ a: 1, b: 1, c: 1 }) const map2 = Map({ a: 1, b: 1, c: 1 }) map1 === map2 //false Object.is(map1, map2) // false is(map1, map2) // true
List() 和 Map()
做用:用来建立一个新的List/Map对象
用法:
//List List(): List<any> List<T>(): List<T> //Map Map(): Map<any> Map<T>(): Map<T>
List.of() 和 Map.of()
做用:建立一个新的包含value的List/Map对象
用法:
List.of<T>(...values: Array<T>): List<T> Map.of<T>(...values: Object<T>): Map<T>
List.isList() 和 Map.isMap()
做用:判断一个数据结构是否是List/Map类型
用法:
List.isList(maybeList: any): boolean Map.isMap(maybeMap: any): boolean
size
做用:获取List/Map的长度
get() 、 getIn()
做用:获取数据结构中的数据
has() 、 hasIn()
做用:判断是否存在某一个key
用法:
Immutable.fromJS([1,2,3,{a:4,b:5}]).has('0'); //true Immutable.fromJS([1,2,3,{a:4,b:5}]).has('0'); //true Immutable.fromJS([1,2,3,{a:4,b:5}]).hasIn([3,'b']) //true
includes()
做用:判断是否存在某一个value
用法:
Immutable.fromJS([1,2,3,{a:4,b:5}]).includes(2); //true Immutable.fromJS([1,2,3,{a:4,b:5}]).includes('2'); //false 不包含字符2 Immutable.fromJS([1,2,3,{a:4,b:5}]).includes(5); //false Immutable.fromJS([1,2,3,{a:4,b:5}]).includes({a:4,b:5}) //false Immutable.fromJS([1,2,3,{a:4,b:5}]).includes(Immutable.fromJS({a:4,b:5})) //true
first() 、 last()
做用:用来获取第一个元素或者最后一个元素,若没有则返回undefined
代码:
Immutable.fromJS([1,2,3,{a:4,b:5}]).first()//1 Immutable.fromJS([1,2,3,{a:4,b:5}]).last()//{a:4,b:5} Immutable.fromJS({a:1,b:2,c:{d:3,e:4}}).first() //1 Immutable.fromJS({a:1,b:2,c:{d:3,e:4}}).first() //{d:3,e:4}
注:这里对于数据的修改,是对原数据进行操做后的值赋值给一个新的数据,并不会对原数据进行修改,由于Immutable是不可变的数据类型。
设置 set()
做用:设置第一层key、index的值
用法:
set(index: number, value: T): List<T> set(key: K, value: V): this
List在使用的时候,将index为number值设置为value。Map在使用的时候,将key的值设置为value。
在List中使用时,若传入的number为负数,则将index为size+index的值设置为value,例,若传入-1,则将size-1的值设为value。若传入的number的值超过了List的长度,则将List自动补全为传入的number的值,将number设置为value,其他用undefined补全。注:跟js中不一样,List中不存在空位,[,,,],List中若没有值,则为undefined。
代码实现:
//////List const originalList = List([ 0 ]); // List [ 0 ] originalList.set(1, 1); // List [ 0, 1 ] originalList.set(0, 'overwritten'); // List [ "overwritten" ] originalList.set(2, 2); // List [ 0, undefined, 2 ] List().set(50000, 'value').size; // 50001 //////Map const { Map } = require('immutable') const originalMap = Map() const newerMap = originalMap.set('key', 'value') const newestMap = newerMap.set('key', 'newer value') originalMap // Map {} newerMap // Map { "key": "value" } newestMap // Map { "key": "newer value" }
setIn()
做用:设置深层结构中某属性的值
用法:
setIn(keyPath: Iterable<any>, value: any): this
用法与set()同样,只是第一个参数是一个数组,表明要设置的属性所在的位置
删除 delete
做用:用来删除第一层结构中的属性
用法:
delete(index: number): List<T> //List delete(key: K): this //Map
deleteIn()
用来删除深层数据,用法参考setIn
deleteAll() (Map独有,List没有)
做用:用来删除Map中的多个key
用法:deleteAll(keys: Iterable<K>): this
代码示例:
const names = Map({ a: "Aaron", b: "Barry", c: "Connor" }) names.deleteAll([ 'a', 'c' ]) // Map { "b": "Barry" }
更新 update()
做用:对对象中的某个属性进行更新,可对原数据进行相关操做
用法:
update(index: number, updater: (value: T) => T): this //List update(key: K, updater: (value: V) => V): this //Map
代码示例:
////List const list = List([ 'a', 'b', 'c' ]) const result = list.update(2, val => val.toUpperCase()) ///Map const aMap = Map({ key: 'value' }) const newMap = aMap.update('key', value => value + value)
updateIn()
用法参考setIn
清除 clear()
做用:清除全部数据
用法:clear(): this
代码示例:
Map({ key: 'value' }).clear() //Map List([ 1, 2, 3, 4 ]).clear() // List
List对应的数据结构是js中的数组,因此数组的一些方法在Immutable中也是通用的,好比push,pop,shift,unshift,insert。
代码实现:
List([ 0, 1, 2, 3, 4 ]).insert(6, 5) //List [ 0, 1, 2, 3, 4, 5 ] List([ 1, 2, 3, 4 ]).push(5) // List [ 1, 2, 3, 4, 5 ] List([ 1, 2, 3, 4 ]).pop() // List[ 1, 2, 3 ] List([ 2, 3, 4]).unshift(1); // List [ 1, 2, 3, 4 ] List([ 0, 1, 2, 3, 4 ]).shift(); // List [ 1, 2, 3, 4 ]
List中还有一个特有的方法用法设置List的长度,setSize()
List([]).setSize(2).toJS() //[undefined,undefined]
merge
做用:浅合并,新数据与旧数据对比,旧数据中不存在的属性直接添加,就数据中已存在的属性用新数据中的覆盖
mergrWith
做用:自定义浅合并,可自行设置某些属性的值
mergeIn
做用:对深层数据进行浅合并
mergeDeep
做用:深合并,新旧数据中同时存在的的属性为新旧数据合并以后的数据
mergeDeepIn
做用:对深层数据进行深合并
mergrDeepWith
做用:自定义深合并,可自行设置某些属性的值
这里用一段示例完全搞懂merge,此示例为Map结构,List与Map原理相同
const Map1 = Immutable.fromJS({a:111,b:222,c:{d:333,e:444}}); const Map2 = Immutable.fromJS({a:111,b:222,c:{e:444,f:555}}); const Map3 = Map1.merge(Map2); //Map {a:111,b:222,c:{e:444,f:555}} const Map4 = Map1.mergeDeep(Map2); //Map {a:111,b:222,c:{d:333,e:444,f:555}} const Map5 = Map1.mergeWith((oldData,newData,key)=>{ if(key === 'a'){ return 666; }else{ return newData } },Map2); //Map {a:666,b:222,c:{e:444,f:555}}
concat()
做用:对象的拼接,用法与js数组中的concat()相同,返回一个新的对象。
用法:const List = list1.concat(list2)
map()
做用:遍历整个对象,对Map/List元素进行操做,返回一个新的对象。
用法:
Map({a:1,b:2}).map(val=>10*val) //Map{a:10,b:20}
Map特有的mapKey()
做用:遍历整个对象,对Map元素的key进行操做,返回一个新的对象。
用法:
Map({a:1,b:2}).mapKey(val=>val+'l') //Map{al:10,bl:20}
Map特有的mapEntries()
做用:遍历整个对象,对Map元素的key和value同时进行操做,返回一个新的对象。Map的map()也可实现此功能。
用法:
Map({a:1,b:2}).map((key,val)=>{ return [key+'l',val*10] }) //Map{al:10,bl:20}
过滤 filter
做用:返回一个新的对象,包括全部知足过滤条件的元素
用法:
Map({a:1,b:2}).filter((key,val)=>{ return val == 2 }) //Map{b:2}
还有一个filterNot()方法,与此方法正好相反。
反转 reverse
做用:将数据的结构进行反转
代码示例:
Immutable.fromJS([1, 2, 3, 4, 5]).reverse(); // List [5,4,3,2,1] Immutable.fromJS({a:1,b:{c:2,d:3},e:4}).recerse(); //Map {e:4,b:{c:2,d:3},a:1}
排序 sort & sortBy
做用:对数据结构进行排序
代码示例:
///List Immutable.fromJS([4,3,5,2,6,1]).sort() // List [1,2,3,4,5,6] Immutable.fromJS([4,3,5,2,6,1]).sort((a,b)=>{ if (a < b) { return -1; } if (a > b) { return 1; } if (a === b) { return 0; } }) // List [1,2,3,4,5,6] Immutable.fromJS([{a:3},{a:2},{a:4},{a:1}]).sortBy((val,index,obj)=>{ return val.get('a') },(a,b)=>{ if (a < b) { return -1; } if (a > b) { return 1; } if (a === b) { return 0; } }) //List [ {a:3}, {a:2}, {a:4}, {a:1} ] //Map Immutable.fromJS( {b:1, a: 3, c: 2, d:5} ).sort() //Map {b: 1, c: 2, a: 3, d: 5} Immutable.fromJS( {b:1, a: 3, c: 2, d:5} ).sort((a,b)=>{ if (a < b) { return -1; } if (a > b) { return 1; } if (a === b) { return 0; } }) //Map {b: 1, c: 2, a: 3, d: 5} Immutable.fromJS( {b:1, a: 3, c: 2, d:5} ).sortBy((value, key, obj)=> { return value }) //Map {b: 1, c: 2, a: 3, d: 5}
分组 groupBy
做用:对数据进行分组
const listOfMaps = List([ Map({ v: 0 }), Map({ v: 1 }), Map({ v: 1 }), Map({ v: 0 }), Map({ v: 2 }) ]) const groupsOfMaps = listOfMaps.groupBy(x => x.get('v')) // Map { // 0: List [ Map{ "v": 0 }, Map { "v": 0 } ], // 1: List [ Map{ "v": 1 }, Map { "v": 1 } ], // 2: List [ Map{ "v": 2 } ], // }
indexOf() 、 lastIndexOf Map不存在此方法
做用:和js数组中的方法相同,查找第一个或者最后一个value的index值,找不到则返回-1
用法:
Immutable.fromJS([1,2,3,4]).indexof(3) //2 Immutable.fromJS([1,2,3,4]).lastIndexof(3) //2
findIndex() 、 findLastIndex() Map不存在此方法
做用:查找知足要求的元素的index值
用法:
Immutable.fromJS([1,2,3,4]).findIndex((value,index,array)=>{ return value%2 === 0; }) // 1 Immutable.fromJS([1,2,3,4]).findLastIndex((value,index,array)=>{ return index%2 === 0; }) // 3
find() 、 findLast()
做用:查找知足条件的元素的value值
用法:
Immutable.fromJS([1,2,3,4]).find((value,index,array)=>{ return value%2 === 0; }) // 2 Immutable.fromJS([1,2,3,4]).findLast((value,index,array)=>{ return value%2 === 0; }) // 4
findKey() 、 findLastKey()
做用:查找知足条件的元素的key值
用法:
Immutable.fromJS([1,2,3,4]).findKey((value,index,array)=>{ return value%2 === 0; }) // 1 Immutable.fromJS([1,2,3,4]).findLastKey((value,index,array)=>{ return value%2 === 0; }) // 3
findEntry() 、 findLastEntry()
做用:查找知足条件的元素的键值对 key:value
用法:
Immutable.fromJS([1,2,3,4]).findEntry((value,index,array)=>{ return value%2 === 0; }) // [1,2] Immutable.fromJS([1,2,3,4]).findLastEntry((value,index,array)=>{ return value%2 === 0; }) // [3,4]
keyOf() lastKeyOf()
做用:查找某一个value对应的key值
用法:
Immutable.fromJS([1,2,3,4]).keyOf(2) //1 Immutable.fromJS([1,2,3,4]).lastKeyOf(2) //1
max() 、 maxBy()
做用:查找最大值
用法:
Immutable.fromJS([1, 2, 3, 4]).max() //4 Immutable.fromJS([{a;1},{a:2},{a: 3},{a:4}]).maxBy((value,index,array)=>{ return value.get('a') }) //{a:4}
min() 、 minBy()
做用:查找最小值
用法:
Immutable.fromJS([1, 2, 3, 4]).min() //1 Immutable.fromJS([{a;1},{a:2},{a: 3},{a:4}]).minBy((value,index,array)=>{ return value.get('a') }) //{a:1}
slice()
做用: 和原生js中数组的slice数组同样,包含两个参数,start和end,start表明开始截取的位置,end表明结束的位置,不包括第end的元素。若不包括end,则返回整个对象,若end为负数,则返回(start,length-end)对应的数据。若start只有一个而且为负数,则返回最后的end个元素。
用法:
Immutable.fromJS([1, 2, 3, 4]).slice(0); //[1,2,3,4] Immutable.fromJS([1, 2, 3, 4]).slice(0,2); //[1,2] Immutable.fromJS([1, 2, 3, 4]).slice(-2); //[3,4] Immutable.fromJS([1, 2, 3, 4]).slice(0,-2); //[1,2]
rest()
做用:返回除第一个元素以外的全部元素
用法:Immutable.fromJS([1, 2, 3, 4]).rest()//[2,3,4]
butLast()
做用:返回除最后一个元素以外的全部元素
用法:Immutable.fromJS([1, 2, 3, 4]).rest()//[1,2,3]
skip()
做用:有一个参数n, 返回截掉前n个元素以后剩下的全部元素
用法:Immutable.fromJS([1, 2, 3, 4]).skip(1)//[2,3,4]
skipLast()
做用:有一个参数n, 返回截掉最后n个元素以后剩下的全部元素
用法:Immutable.fromJS([1, 2, 3, 4]).skip(1)//[1,2,3]
skipWhile()
做用:返回从第一次返回false以后的全部元素
Immutable.fromJS([1, 2, 3, 4]).skipWhile(list.skipWhile((value,index,list)=>{ return value > 2; }))// [1,2,3,4]
skipUntil()
做用:返回从第一次返回true以后的全部元素
Immutable.fromJS([1, 2, 3, 4]).skipUntil(list.skipWhile((value,index,list)=>{ return value > 2; }))// [3,4]
take()
做用:有一个参数n, 返回前n个元素
用法:Immutable.fromJS([1, 2, 3, 4]).take(2)//[1,2]
takeLast()
做用:有一个参数n, 返回最后n个元素
用法:Immutable.fromJS([1, 2, 3, 4]).takeLast(2)//[3,4]
takeWhile()
做用:返回从第一次返回false以前的全部元素
Immutable.fromJS([1, 2, 3, 4]).skipWhile(list.takeWhile((value,index,list)=>{ return value > 2; }))// []
takeUntil()
做用:返回从第一次返回true以前的全部元素
Immutable.fromJS([1, 2, 3, 4]).skipUntil(list.takeUntil((value,index,list)=>{ return value > 2; }))// [1,2]
reduce()
做用:和js中数组中的reduce相同,按索引升序的顺序处理元素
用法:
Immutable.fromJS([1,2,3,4]).reduce((pre,next,index,arr)=>{ console.log(pre+next) return pre+next; }) // 3 6 10
reduceRight()
做用:和js中数组中的reduce相同,按索引降序的顺序处理元素
用法:
Immutable.fromJS([1,2,3,4]).reduceRight((pre,next,index,arr)=>{ console.log(pre+next) return pre+next; }) // 7 9 10
every()
做用:判断整个对象总中全部的元素是否是都知足某一个条件,都知足返回true,反之返回false。
代码:
Immutable.fromJS([1,2,3,4]).every((value,index,arr)=>{ return value > 2 }) // false
some()
做用:判断整个对象总中全部的元素是否是存在知足某一个条件的元素,若存在返回true,反之返回false。
代码:
Immutable.fromJS([1,2,3,4]).some((value,index,arr)=>{ return value > 2 }) // true
join()
做用:同js中数组的join方法。把准换为字符串
用法:Immutable.fromJS([1,2,3,4]).join(',') //1,2,3,4
isEmpty()
做用:判断是否为空
用法:
Immutable.fromJS([]).isEmpty(); // true Immutable.fromJS({}).isEmpty(); // true
count()
做用:返回元素个数,可自定义条件,返回知足条件的个数
用法:
const list = Immutable.fromJS([1,2,3,4]); const map = Immutable.fromJS({a:1,b:2,c:3,d:4}); list.count((value,index,list)=>{ return value > 2; }) //2 map.count((value,index,list)=>{ return value > 2; }) //2
countBy()
做用:与count不一样的是,countBy返回一个对象
用法:
const list = Immutable.fromJS([1,2,3,4]); const map = Immutable.fromJS({a:1,b:2,c:3,d:4}); list.countBy((value,index,list)=>{ return value > 2; } //{false: 2, true: 2} map.countBy((value,index,list)=>{ return value > 2; } //{false: 2, true: 2}