JavaScript 中的对象一般是可变的,因为Javascript为了节省内存,都是使用引用赋值,生成一个新对象如果简单的引用了原始对象,那对新对象作出的改变,也将影响到原始对象。平时简单的应用还不会特别明显,但是如果在复杂的应用中,对象的改变就会引起比较复杂的问题。往往都会使用deepCopy来解决这个问题,但是deepCopy往往会对整个对象的树进行拷贝,也会造成 CPU 和内存的浪费。
一、什么是immutable data?
Immutable Data 就是被创建之后不能给改变数据。Immutable 实现的原理是 Persistent Data Structure(持久化数据结构),新数据通过旧的数据创建,需要使旧数据依旧可用而且不能被改变。Immutable的这种实现原理,也避免了deepCopy复制所有节点带来的性能损耗。Immutable的实现原理见下图:

二、Immutable 优点
1. Immutable 降低了 Mutable 带来的复杂度
2. 节省内存
3. Undo/Redo,Copy/Paste,甚至时间旅行这些功能做起来小菜一碟
4. 并发安全
5. 拥抱函数式编程

三、Immutable 缺点:
1. 需要学习新的 API
2. 增加了资源文件大小
3. 容易与原生对象混淆

四、Immutable的数据类型
Immutable 的几种数据类型
1.List: 有序索引集,类似JavaScript中的Array。
2.Map: 无序索引集,类似JavaScript中的Object。
3.OrderedMap: 有序的Map,根据数据的set()进行排序。
4.Set: 没有重复值的集合。
5.OrderedSet: 有序的Set,根据数据的add进行排序。
6.Stack: 有序集合,支持使用unshift()和shift()添加和删除。
7.Range(): 返回一个Seq.Indexed类型的集合,这个方法有三个参数,start表示开始值,默认值为0,end表示结束值,默认为无穷大,step代表每次增大的数值,默认为1.如果start = end,则返回空集合。
8.Repeat(): 返回一个vSeq.Indexe类型的集合,这个方法有两个参数,value代表需要重复的值,times代表要重复的次数,默认为无穷大。
9.Record: 一个用于生成Record实例的类。类似于JavaScript的Object,但是只接收特定字符串为key,具有默认值。
10.Seq: 序列,但是可能不能由具体的数据结构支持。
11.Collection: 是构建所有数据结构的基类,不可以直接构建。

用的最多就是List和Map,所以在这里主要介绍这两种数据类型的API。

五、常用的API
1.fromJS() 将一个js数据转换为Immutable类型的数据。
用法:fromJS(value, converter)
简介:value是要转变的数据,converter是要做的操作。第二个参数可不填,默认情况会将数组准换为List类型,将对象转换为Map类型,其余不做操作。

const obj = Immutable.fromJS({a:'123',b:'234'},function (key, value, path) {console.log(key, value, path)return isIndexed(value) ? value.toList() : value.toOrderedMap())
})

2.toJS()
作用:将一个Immutable数据转换为JS类型的数据。
用法:value.toJS()

3.is()对两个对象进行比较。
用法: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

4.List 和 Map
创建
List() 和 Map()

作用:用来创建一个新的List/Map对象

用法:

//ListList(): List<any>
List<T>(): List<T>//MapMap(): 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): booleanMap.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中的各种删除与插入
List对应的数据结构是js中的数组,所以数组的一些方法在Immutable中也是通用的,比如push,pop,shift,unshift,insert。

push():在List末尾插入一个元素
pop(): 在List末尾删除一个元素
unshift: 在List首部插入一个元素
shift: 在List首部删除一个元素
insert:在List的index处插入元素
代码实现:

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
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} ]

//MapImmutable.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;
})  // 2Immutable.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;
})  // 1Immutable.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() //4Immutable.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() //1Immutable.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;
})    //2map.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}

immutable学习相关推荐

  1. 【immutable学习记录】immutable 中 Seq 的用法

    文章目录 什么是immutable Seq 的介绍及 API 什么是immutable Immutable Data 就是一旦创建,就不能再被更改的数据.对 Immutable 对象的任何修改或添加删 ...

  2. 用 React+Redux+Immutable 做俄罗斯方块

    俄罗斯方块是一直各类程序语言热衷实现的经典游戏,JavsScript的实现版本也有很多,用React 做好俄罗斯方块则成了我一个目标. 戳 https://chvin.github.io/react- ...

  3. 07-SparkStreaming

    07-Spark Streaming 1.目录概述 掌握spark Streaming的原理和架构 掌握DStream的相关操作 实现spark Streaming与flume整合 实现spark S ...

  4. python学习-元组的复习及进阶使用(iterable、indexing and slicing、sequence unpacking、immutable、named tuple)

    文章目录 可迭代的(iterable) 索引和切片(indexing and slicing) 序列解包(sequence unpacking) 不可修改的(immutable) named tupl ...

  5. 你确实应该学习并使用的 10 个 C# 特性

    转自:http://blog.jobbole.com/99599/ 1) async / await 使用 async / await 模式,可以在执行代码块操作的时候不会阻塞 UI 或者当前的线程. ...

  6. 前端最新技术学习地址总结

    2019独角兽企业重金招聘Python工程师标准>>> 1. 主流技术官网总结 1.1. 工具类 Glup 官网:http://www.gulpjs.com.cn/ http://i ...

  7. 辛湜推荐的数据库领域的一些学习材料

    之前林仕鼎曾整理过系统架构领域的学习资料,这几天Spark核心团队成员辛湜(Reynold Xin)公开了他整理的一份数据库学习资料列表,很有价值,Hacker News上引起了不少讨论.简要编译如下 ...

  8. 2017 年该学习的编程语言、框架和工具

    转:http://top.jobbole.com/35926/?utm_source=blog.jobbole.com&utm_medium=sidebar-top-news 软件开发行业继续 ...

  9. JavaScript学习总结(一)——ECMAScript、BOM、DOM(核心、浏览器对象模型与文档对象模型)...

    一.JavaScript简介 JavaScript是一种解释执行的脚本语言,是一种动态类型.弱类型.基于原型的语言,内置支持类型,它遵循ECMAScript标准.它的解释器被称为JavaScript引 ...

最新文章

  1. 2021年2月反思与总结
  2. Google工程师:如何看待程序员普遍缺乏数据结构和算法知识?
  3. Java8函数式编程(2)--流与管道
  4. java解决错误经验_在Java错误进入生产之前的新处理方式
  5. kudu:插入报错:Row error for primary key=9, tablet=null, server=xx, status=Not found: key not found
  6. sql视图能使用触发器吗_白刚玉粉末能重复使用吗
  7. 轮子来袭 vJine.Core Orm 之 01_快速体验
  8. python读取数据库后生成网页_python查询数据库并将结果按照格式生成HTML页面展示...
  9. v4手游服务器维护,v4手游每日必做事项分享
  10. Rust的crate
  11. 学习笔记42—Win7下安装Linux双系统
  12. 【uni-app】懂你找图--创建项目到首页推荐模块
  13. 如何快速学会一项新技能?
  14. 什么是C标签 为什么要用C 标签
  15. Matlab高光谱遥感、数据处理与混合像元分解实践技术应用
  16. 刻意学习:持续行动让你人生逆袭
  17. Web应用优化 - 优化切页模式
  18. bootloader的两种启动模式
  19. 【UI】锤子手机-坚果手机-文艺青年版-配色色号
  20. 50个最佳机器学习公共数据集

热门文章

  1. 外贸人看过来|Get 各国谈判风格
  2. 如何有更好的职业发展
  3. 互动媒体技术——使用processing代码生成花
  4. html框架布局模板上下结构的字有哪些,网站常用布局类型格式
  5. Launcher3桌面开发(5)-Launcher3 负一屏的定制开发
  6. 学生党开学收藏榜单,通话质量好的蓝牙耳机
  7. OPPO Reno6的charles证书
  8. ThinkPHP6 think-queue 使用
  9. btree高度mysql_如何知道btree树的高度
  10. python画大象_Python Day8