JavaScript 函数式编程读书笔记2
概述
这是我读《javascript函数式编程》的读书笔记,供以后开发时参考,相信对其他人也有用。
说明:虽然本书是基于underscore.js库写的,但是其中的理念和思考方式都讲的很好,值得一读。不过如果不熟悉underscore.js库的话,读起来会有点困难。
对象的不变性
函数式编程中函数是一等公民,所以对于数据来说,需要他们是不可变的。另外,对象的不可变确实能够带来一些好处,比如:如果数据是不可变的,那么可以直接通过“===”来判断数据是否发生变化,这使得react的效率得到极大提升。下面我们来讨论在js中实现immutable对象的方法。
freeze
用Object.freeze可以冻结一个数组或者对象,当使用Object.freeze的时候,将导致后续的变化失败。如果在严格模式下使用,将抛出一个TypeError;否则,所有的变化将悄悄地失败。实例如下:
var a = [2,3,4];
a[1]; //输出3
Object.freeze(a);
a[1]=5;
a[1]; //输出3
也有一个isFrozen API来判断a是否确实被冻结。
Object.isFrozen(a); //输出true
但是这么实现有如下几个痛点:
- 变化会悄悄地失败,可能会导致微妙的错误。
 - 不是所有的库都支持freeze。
 - 这个是最重要的,Object.freeze是一个浅操作。
 
由于Object.freeze是一个浅操作,所以嵌套对象里面的变量不会被冻结,除非用递归进行深冻结,代码如下:
function isObject(obj) {
    var type = typeof obj;
    return type === 'function' || type === 'object' && !!obj;
}
function deepFreeze(obj) {
    if(!Object.isFrozen(obj)) Object.freeze(obj);
    for (var key in obj) {
        if(!obj.hasOwnProperty(key) || !isObject(obj)) continue;
        deepFreeze(obj[key]);
    }
}
使用容器
我们也可以使用深复制,并把它放在一个容器里面来保持变量的不变性。简单代码如下:
function isObject(obj) {
    var type = typeof obj;
    return type === 'function' || type === 'object' && !!obj;
}
function deepClone(obj) {
    if(!isObject(obj)) return obj;
    var temp = new obj.constructor();
    for (var key in obj) {
        if(obj.hasOwnProperty(key))
            temp[key] = deepClone(obj[key]);
    }
    return temp;
}
function Container(init) {
    this.state = init;
}
Container.prototype = {
    setState: function(obj) {
        this.state = deepClone(obj);
        return this.state;
    }
}
var aObj = new Container({a:{b:1}, c:2});
aObj.setState({a:{b:3}, c:6, d:8});
aObj; //输出{a:{b:3}, c:6, d:8}
使用immutable.js
使用上面的方法实现不可变数据问题很多,用freeze会悄悄地失败,并且是浅操作,使用容器的话深复制对象会消耗大量的内存,并且速度很慢。
immutable.js就是facebook开源的为了解决这个问题的库。它可以说是和react一样的划时代的产物。它通过参考hash maps tries和hash maps tries提供了一种优雅地实现javascript Immutable Data的方式。
它使用了Structural Sharing(结构共享),即如果对象树中一个节点发生变化,只修改这个节点和受它影响的父节点,其它节点则进行共享。所以避免了深拷贝带来的CPU 和内存的浪费。
具体可参考:
搞定immutable.js详细说明
IMMUTABLE 详解
JavaScript 函数式编程读书笔记2的更多相关文章
- JavaScript 函数式编程读书笔记1
		
概述 这是我读<javascript函数式编程>的读书笔记,供以后开发时参考,相信对其他人也有用. 说明:虽然本书是基于underscore.js库写的,但是其中的理念和思考方式都讲的很好 ...
 - Node.js高级编程读书笔记Outline
		
Motivation 世俗一把,看看前端的JavaScript究竟能做什么. 顺便检验一下自己的学习能力. Audience 想看偏后台的Java程序员关于前端JavaScript的认识的职业前端工程 ...
 - 转:JavaScript函数式编程(三)
		
转:JavaScript函数式编程(三) 作者: Stark伟 这是完结篇了. 在第二篇文章里,我们介绍了 Maybe.Either.IO 等几种常见的 Functor,或许很多看完第二篇文章的人都会 ...
 - 转: JavaScript函数式编程(二)
		
转: JavaScript函数式编程(二) 作者: Stark伟 上一篇文章里我们提到了纯函数的概念,所谓的纯函数就是,对于相同的输入,永远会得到相同的输出,而且没有任何可观察的副作用,也不依赖外部环 ...
 - 转:JavaScript函数式编程(一)
		
转:JavaScript函数式编程(一) 一.引言 说到函数式编程,大家可能第一印象都是学院派的那些晦涩难懂的代码,充满了一大堆抽象的不知所云的符号,似乎只有大学里的计算机教授才会使用这些东西.在曾经 ...
 - python高级编程读书笔记(一)
		
python高级编程读书笔记(一) python 高级编程读书笔记,记录一下基础和高级用法 python2和python3兼容处理 使用sys模块使程序python2和python3兼容 import ...
 - C++Windows核心编程读书笔记
		
转自:http://www.makaidong.com/%E5%8D%9A%E5%AE%A2%E5%9B%AD%E6%96%87/71405.shtml "C++Windows核心编程读书笔 ...
 - 一文带你了解JavaScript函数式编程
		
摘要: 函数式编程入门. 作者:浪里行舟 Fundebug经授权转载,版权归原作者所有. 前言 函数式编程在前端已经成为了一个非常热门的话题.在最近几年里,我们看到非常多的应用程序代码库里大量使用着函 ...
 - javascript函数式编程和链式优化
		
1.函数式编程理解 函数式编程可以理解为,以函数作为主要载体的编程方式,用函数去拆解.抽象一般的表达式 与命令式相比,这样做的好处在哪?主要有以下几点: (1)语义更加清晰 (2)可复用性更高 (3) ...
 
随机推荐
- mysql实用函数
			
1. group_concat(); 可以将选择的字段列数据,分组以逗号分隔成一串.实用方便.select id,group_concat(distinct name) from ttt group ...
 - 如何关闭wps热点,如何关闭wpscenter,如何关闭我的wps
			
用wps已经快十年了,最开始的时候速度快,非常好用,甩office几条街,但最近这几年随着wps胃口越来越大,各种在线功能不断推出,植入广告越来越多,逐渐让人失去欢喜. 通过各种网帖的经验,我把网上流 ...
 - 关于Shader的学习记录
			
float4 _EmissiveColor; float4 _AmbientColor; float _MySliderValue; void surf (Input IN, inout Surfac ...
 - jQuery之必会增删改查Dom操作
			
.next .prev <button>change</button> <span class = '.demo'>aaa</span> <p ...
 - uboot1.1.6
			
http://blog.csdn.net/lizuobin2/article/details/52061530
 - Java比较两个时间的前后
			
public static int compare_date(String DATE1, String DATE2) { DateFormat df = new SimpleDateFormat(&q ...
 - [leetcode]364. Nested List Weight Sum II嵌套列表加权和II
			
Given a nested list of integers, return the sum of all integers in the list weighted by their depth. ...
 - Linux-目录结构及文件系统
			
1.Linux 系统的顶层目录结构 / 根目录 ├── bin 存放用户二进制文件 ├── boot 存放内核引导配置文件 ├── dev 存放设备文件 ...
 - Elasticsearch tshark 封包分析 (转)
			
Elasticsearch tshark 封包分析 使用wireshark能解決許多網路問題,將側錄下來的封包傳至Elasticsearch上方便分析製作及時報表.tshark為wireshark的命 ...
 - dbus 消息和消息总线实例讲解-一
			
应用程序A和消息总线连接,这个连接获取了一个众所周知的公共名(记作连接A).应用程序A中有对象A1提供了接口I1,接口I1有方法M1. 应用程序B和消息总线连接,要求调用连接A上对象A1的接口I1的方 ...