[译] 利用 Immutability(不可变性)编写更为简洁高效的代码

栏目: Node.js · 发布时间: 6年前

内容简介:[译] 利用 Immutability(不可变性)编写更为简洁高效的代码
[译] 利用 Immutability(不可变性)编写更为简洁高效的代码

图片来自https://unsplash.com

不可变性是函数式编程中的一部分,它可以使你写出更安全更简洁的代码。我将会通过一些 JavaScript 的例子来告诉你如何达到不可变性。

根据维基(地址 ):

一个不可变对象(不能被改变的对象)是指在创建之后其状态不能被更改的对象,这与在创建之后可以被更改的可变对象(可以被改变的对象)相反。在某些情况下,一个对象的外部状态如果从外部看来没有变化,那么即使它的一些内部属性更改了,仍被视为不可变对象。

不可变的数组

数组是了解不可变性如何运作的一个很好的起点。我们来看一下。

const arrayA = [1, 2, 3];
arrayA.push(4);

const arrayB = arrayA;
arrayB.push(5);

console.log(arrayA); // [1, 2, 3, 4, 5]
console.log(arrayB); // [1, 2, 3, 4, 5]

例子中 arrayBarrayA 的引用,所以如果我们通过 push 方法向任意数组中添加一个值 5,那么就会间接影响到另外一个,这个是违反不可变性的原则的。

我们可以通过使用slice 函数以达到不可变性,进而优化我们的例子,此时代码的行为是完全不一样的。

const arrayA = [1, 2, 3];
arrayA.push(4);

const arrayB = arrayA.slice(0);
arrayB.push(5);

console.log(arrayA); // [1, 2, 3, 4]
console.log(arrayB); // [1, 2, 3, 4, 5]

这才是我们要的,代码不改变其它的值。

记住:当使用push 来给数组添加一个值时,你在 改变 这个数组,因为这样可能会影响代码里的其他部分,所以你想要避免使变量值发生改变。slice 会返回一个复制的数组。

函数

现在你知道了如何避免改变其它的值。那如何写「纯」的函数呢?纯函数是指不会产生任何副作用,也不会改变状态的函数。

我们来看一个示例函数,其原理与前面数组示例的原理相同。首先我们写一个会改变其它值的函数,然后我们将这个函数优化为「纯」函数。

const add = (arrayInput, value) => {
  arrayInput.push(value);

  return arrayInput;
};
const array = [1, 2, 3];

console.log(add(array, 4)); // [1, 2, 3, 4]
console.log(add(array, 5)); // [1, 2, 3, 4, 5]

于是我们又一次 改变 输入的变量的值,这使得这个函数变得不可预测。在函数式编程的世界里,有一个关于函数的铁律: 函数对于相同的输入应当返回相同的值。

上面的函数违反了这一规则,每次我们调用 add 方法,它都会改变 数组 变量导致结果不一样。

让我们来看看怎样修改 add 函数来使其不可变。

const add = (arrayInput, value) => {
  const copiedArray = arrayInput.slice(0);
  copiedArray.push(value);

  return copiedArray;
};

const array = [1, 2, 3];
const resultA = add(array, 4);
console.log(resultA); // [1, 2, 3, 4]
const resultB = add(array, 5);
console.log(resultB); // [1, 2, 3, 5]

现在我们可以多次调用这个函数,且相同的输入获得相同的输出,与预期一致。这是因为我们不再改变 array 变量。我们把这个函数叫做“纯函数”。

注意:你还可以使用 concat ,来代替 slicepush

即:arrayInput.concat(value);

我们还可以使用 ES6 的扩展语法,来简化函数。

const add = (arrayInput, value) => […arrayInput, value];

并发

NodeJS 的应用有一个叫并发的概念,并发操作是指两个计算可以同时的进行而不用管另外的一个。如果有两个线程,第二个计算不需要等待第一个完成即可开始。

[译] 利用 Immutability(不可变性)编写更为简洁高效的代码

可视化的并发操作

NodeJS 用事件循环机制使并发成为可能。事件循环重复接收事件,并一次触发一个监听该事件的处理程序。这个模型允许 NodeJS 的应用处理大规模的请求。如果你想学习更多,读一下 这篇关于事件循环的文章

不可变性跟并发又有什么关系呢?由于多个操作可能会并发地改变函数的作用域的值,这将会产生不可靠的输出和导致意想不到的结果。注意函数是否改变它作用域之外的值,因为这可能真的会很危险。

下一步

不可变性是学习函数式编程过程中的一个重要概念。你可以了解一下由 Facebook 开发者写的ImmutableJS,这一个库提供一些不可变的数据结构,比如说 MapSet 、和 List

[译] 利用 Immutability(不可变性)编写更为简洁高效的代码

点击 :blue_heart: 让更多的人可以在 Medium 上看见这篇文章,感谢阅读。

掘金翻译计划 是一个翻译优质互联网技术文章的社区,文章来源为掘金 上的英文分享文章。内容覆盖 AndroidiOSReact前端后端产品设计 等领域,想要查看更多优质译文请持续关注 掘金翻译计划


以上所述就是小编给大家介绍的《[译] 利用 Immutability(不可变性)编写更为简洁高效的代码》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

架构即未来:现代企业可扩展的Web架构、流程和组织(原书第2版)

架构即未来:现代企业可扩展的Web架构、流程和组织(原书第2版)

Martin L. Abbott、Michael T. Fisher / 陈斌 / 机械工业出版社 / 2016-4-15 / 99.00

任何一个持续成长的公司最终都需要解决系统、组织和流程的扩展性问题。本书汇聚了作者从eBay、VISA、Salesforce.com到Apple超过30年的丰富经验, 全面阐释了经过验证的信息技术扩展方法,对所需要掌握的产品和服务的平滑扩展做了详尽的论述,并在第1版的基础上更新了扩展的策略、技术和案例。 针对技术和非技术的决策者,马丁•阿伯特和迈克尔•费舍尔详尽地介绍了影响扩展性的各个方面,包......一起来看看 《架构即未来:现代企业可扩展的Web架构、流程和组织(原书第2版)》 这本书的介绍吧!

JS 压缩/解压工具
JS 压缩/解压工具

在线压缩/解压 JS 代码

正则表达式在线测试
正则表达式在线测试

正则表达式在线测试