iis服务器助手广告广告
返回顶部
首页 > 资讯 > 前端开发 > JavaScript >什么是JavaScript函数式编程
  • 525
分享到

什么是JavaScript函数式编程

2024-04-02 19:04:59 525人浏览 安东尼
摘要

这篇文章主要讲解了“什么是javascript函数式编程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“什么是JavaScript函数式编程”吧!JavaSc

这篇文章主要讲解了“什么是javascript函数式编程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“什么是JavaScript函数式编程”吧!

JavaScript 函数式编程是一个存在了很久的话题,但似乎从 2016 年开始,它变得越来越火热。这可能是因为 es6 语法对于函数式编程更为友好,也可能是因为诸如 Rxjs (ReactiveX) 等函数式框架的流行。

看过许多关于函数式编程的讲解,但是其中大部分是停留在理论层面,还有一些是仅针对 Haskell 等纯函数式编程语言的。而本文旨在聊一聊我眼中的函数式编程在 JavaScript 中的具体实践,之所以是 “我眼中的” 即我所说的仅代表个人观点,可能和部分 严格概念 是有冲突的。

本文将略去一大堆形式化的概念介绍,重点展示在 JavaScript 中到底什么是函数式的代码、函数式代码与一般写法有什么区别、函数式的代码能给我们带来什么好处以及常见的一些函数式模型都有哪些。

我理解的函数式编程

我认为函数式编程可以理解为,以函数作为主要载体的编程方式,用函数去拆解、抽象一般的表达式

与命令式相比,这样做的好处在哪?主要有以下几点:

  • 语义更加清晰

  • 可复用性更高

  • 可维护性更好

  • 作用域局限,副作用少

基本的函数式编程

下面例子是一个具体的函数式体现

// 数组中每个单词,首字母大写// 一般写法const arr = ['apple', 'pen', 'apple-pen'];for(const i in arr){  const c = arr[i][0];
  arr[i] = c.toUpperCase() + arr[i].slice(1);
}console.log(arr);// 函数式写法一function upperFirst(Word) {  return word[0].toUpperCase() + word.slice(1);
}function wordToUpperCase(arr) {  return arr.map(upperFirst);
}console.log(wordToUpperCase(['apple', 'pen', 'apple-pen']));// 函数式写法二console.log(arr.map(['apple', 'pen', 'apple-pen'], word => word[0].toUpperCase() + word.slice(1)));

当情况变得更加复杂时,表达式的写法会遇到几个问题:

  1. 鸿蒙官方战略合作共建——HarmonyOS技术社区

  2. 表意不明显,逐渐变得难以维护

  3. 复用性差,会产生更多的代码量

  4. 会产生很多中间变量

函数式编程很好的解决了上述问题。首先参看 函数式写法一,它利用了函数封装性将功能做拆解(粒度不***),并封装为不同的函数,而再利用组合的调用达到目的。这样做使得表意清晰,易于维护、复用以及扩展。其次利用 高阶函数Array.map 代替 for…of 做数组遍历,减少了中间变量和操作。

而 函数式写法一 和 函数式写法二 之间的主要差别在于,可以考虑函数是否后续有复用的可能,如果没有,则后者更优。

链式优化

从上面 函数式写法二 中我们可以看出,函数式代码在写的过程中,很容易造成 横向延展,即产生多层嵌套,下面我们举个比较极端点的例子。

// 计算数字之和

// 一般写法
console.log(1 + 2 + 3 - 4)

// 函数式写法
function sum(a, b) {  return a + b;
}

function sub(a, b) {  return a - b;
}

console.log(sub(sum(sum(1, 2), 3), 4);

本例仅为展示 横向延展 的比较极端的情况,随着函数的嵌套层数不断增多,导致代码的可读性大幅下降,还很容易产生错误。

在这种情况下,我们可以考虑多种优化方式,比如下面的 链式优化 。

// 优化写法 (嗯,你没看错,这就是 lodash 的链式写法)const utils = {
  chain(a) {this._temp = a;return this;
  },
  sum(b) {this._temp += b;return this;
  },
  sub(b) {this._temp -= b;return this;
  },
  value() {const _temp = this._temp;this._temp = undefined;return _temp;
  }
};console.log(utils.chain(1).sum(2).sum(3).sub(4).value());

这样改写后,结构会整体变得比较清晰,而且链的每一环在做什么也可以很容易的展现出来。函数的嵌套和链式的对比还有一个很好的例子,那就是 回调函数 和 Promise 模式

// 顺序请求两个接口// 回调函数import $ from 'Jquery';
$.post('a/url/to/target', (rs) => {  if(rs){
    $.post('a/url/to/another/target', (rs2) => {      if(rs2){
        $.post('a/url/to/third/target');
      }
    });
  }
});// Promiseimport request from 'catta';  // catta 是一个轻量级请求工具,支持 fetch,JSONp,ajax,无依赖request('a/url/to/target')
  .then(rs => rs ? $.post('a/url/to/another/target') : Promise.reject())
  .then(rs2 => rs2 ? $.post('a/url/to/third/target') : Promise.reject());

随着回调函数嵌套层级和单层复杂度增加,它将会变得臃肿且难以维护,而 Promise 的链式结构,在高复杂度时,仍能纵向扩展,而且层次隔离很清晰。

常见的函数式编程模型

闭包(Closure)

可以保留局部变量不被释放的代码块,被称为一个闭包

闭包的概念比较抽象,相信大家都或多或少知道、用到这个特性

那么闭包到底能给我们带来什么好处?

先来看一下如何创建一个闭包:

// 创建一个闭包function makeCounter() {  let k = 0;  return function() {return ++k;
  };
}const counter = makeCounter();console.log(counter());  // 1console.log(counter());  // 2

makeCounter 这个函数的代码块,在返回的函数中,对局部变量 k ,进行了引用,导致局部变量无法在函数执行结束后,被系统回收掉,从而产生了闭包。而这个闭包的作用就是,“保留住“ 了局部变量,使内层函数调用时,可以重复使用该变量;而不同于全局变量,该变量只能在函数内部被引用。

换句话说,闭包其实就是创造出了一些函数私有的 ”持久化变量“。

所以从这个例子,我们可以总结出,闭包的创造条件是:

  1. 鸿蒙官方战略合作共建——HarmonyOS技术社区

  2. 存在内、外两层函数

  3. 内层函数对外层函数的局部变量进行了引用

闭包的用途

闭包的主要用途就是可以定义一些作用域局限的持久化变量,这些变量可以用来做缓存或者计算的中间量等等。

// 简单的缓存工具// 匿名函数创造了一个闭包const cache = (function() {  const store = {};  return {get(key) {      return store[key];
    },set(key, val) {
      store[key] = val;
    }
  }
}());

cache.set('a', 1);
cache.get('a');  // 1

上面例子是一个简单的缓存工具的实现,匿名函数创造了一个闭包,使得 store 对象 ,一直可以被引用,不会被回收。

闭包的弊端

持久化变量不会被正常释放,持续占用内存空间,很容易造成内存浪费,所以一般需要一些额外手动的清理机制。

高阶函数

接受或者返回一个函数的函数称为高阶函数

听上去很高冷的一个词汇,但是其实我们经常用到,只是原来不知道他们的名字而已。JavaScript 语言是原生支持高阶函数的,因为 JavaScript 的函数是一等公民,它既可以作为参数又可以作为另一个函数的返回值使用。

我们经常可以在 JavaScript 中见到许多原生的高阶函数,例如 Array.map , Array.reduce , Array.filter

下面以 map 为例,我们看看他是如何使用的

map (映射)

映射是对集合而言的,即把集合的每一项都做相同的变换,产生一个新的集合

map 作为一个高阶函数,他接受一个函数参数作为映射的逻辑

// 数组中每一项加一,组成一个新数组// 一般写法const arr = [1,2,3];const rs = [];for(const n of arr){
  rs.push(++n);
}console.log(rs)// map改写const arr = [1,2,3];const rs = arr.map(n => ++n);

上面一般写法,利用 for...of 循环的方式遍历数组会产生额外的操作,而且有改变原数组的风险

而 map 函数封装了必要的操作,使我们仅需要关心映射逻辑的函数实现即可,减少了代码量,也降低了副作用产生的风险。

柯里化(Currying)

给定一个函数的部分参数,生成一个接受其他参数的新函数

可能不常听到这个名词,但是用过 undescore 或 lodash 的人都见过他。

有一个神奇的 _.partial 函数,它就是柯里化的实现

// 获取目标文件对基础路径的相对路径// 一般写法const BASE = '/path/to/base';const relativePath = path.relative(BASE, '/some/path');// _.parical 改写const BASE = '/path/to/base';const relativeFromBase = _.partial(path.relative, BASE);const relativePath = relativeFromBase('/some/path');

通过 _.partial ,我们得到了新的函数 relativeFromBase ,这个函数在调用时就相当于调用 path.relative ,并默认将***个参数传入 BASE ,后续传入的参数顺序后置。

本例中,我们真正想完成的操作是每次获得相对于 BASE 的路径,而非相对于任何路径。柯里化可以使我们只关心函数的部分参数,使函数的用途更加清晰,调用更加简单。

组合(Composing)

将多个函数的能力合并,创造一个新的函数

同样你***次见到他可能还是在 lodash 中,compose 方法(现在叫 flow

// 数组中每个单词大写,做 Base64// 一般写法 (其中一种)const arr = ['pen', 'apple', 'applypen'];const rs = [];for(const w of arr){
  rs.push(btoa(w.toUpperCase()));
}console.log(rs);// _.flow 改写const arr = ['pen', 'apple', 'applypen'];const upperAndBase64 = _.partialRight(_.map, _.flow(_.upperCase, btoa));console.log(upperAndBase64(arr));

_.flow 将转大写和转 Base64 的函数的能力合并,生成一个新的函数。方便作为参数函数或后续复用。

自己的观点

我理解的 JavaScript 函数式编程,可能和许多传统概念不同。我并不只认为 高阶函数 算函数式编程,其他的诸如普通函数结合调用、链式结构等,我都认为属于函数式编程的范畴,只要他们是以函数作为主要载体的。

而我认为函数式编程并不是必须的,它也不应该是一个强制的规定或要求。与面向对象或其他思想一样,它也是其中一种方式。我们更多情况下,应该是几者的结合,而不是局限于概念。

感谢各位的阅读,以上就是“什么是JavaScript函数式编程”的内容了,经过本文的学习后,相信大家对什么是JavaScript函数式编程这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

--结束END--

本文标题: 什么是JavaScript函数式编程

本文链接: https://www.lsjlt.com/news/66521.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 什么是JavaScript函数式编程
    这篇文章主要讲解了“什么是JavaScript函数式编程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“什么是JavaScript函数式编程”吧!JavaSc...
    99+
    2024-04-02
  • javascript用函数式编程的原因是什么
    这篇“javascript用函数式编程的原因是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看...
    99+
    2024-04-02
  • Python函数式编程是什么
    这篇文章将为大家详细讲解有关Python函数式编程是什么,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。函数范式在命令式范式中,通过为计算机提供一系列指令然后执行它们来完成任务。在执行这些指令时,可以改变某...
    99+
    2023-06-29
  • 什么是python函数式编程
    python函数式编程是一种抽象程度很高的编程范式,纯粹的函数式编程语言编写的函数没有变量,因此,任意一个函数,只要输入是确定的,输出就是确定的。但python对函数式编程只提供部分支持,且python中允许使用变量,所以python不是不...
    99+
    2024-04-02
  • JavaScript数据类型对函数式编程的影响是什么
    这篇文章主要介绍了JavaScript数据类型对函数式编程的影响是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JavaScript数据类型对函数式编程的影响是什么文章都会有所收获,下面我们一起来看看吧。J...
    99+
    2023-07-05
  • Java8函数式编程方法是什么
    这篇“Java8函数式编程方法是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Java8函数式编程方法是什么”文章吧。什...
    99+
    2023-06-26
  • javascript函数式编程基础
    目录一、引言二、什么是函数式编程三、纯函数(函数式编程的基石,无副作用的函数)四、函数柯里化五、函数组合六、声明式和命令式代码七、Point Free八、示例应用九、总结 一、引言...
    99+
    2024-04-02
  • 用JavaScript实现函数式编程
    这篇文章主要讲解了“用JavaScript实现函数式编程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“用JavaScript实现函数式编程”吧!函数式编程语...
    99+
    2024-04-02
  • javascript函数式编程的用法
    本篇内容介绍了“javascript函数式编程的用法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!说明函数式编程是一种编程范式,是一种软件开...
    99+
    2023-06-20
  • JavaScript 中的函数式编程:为什么它比常规编程更好?
    JavaScript 是一种脚本语言,它被广泛应用于 Web 开发中。在 JavaScript 中,函数式编程已经成为了一种流行的编程风格。而这种编程风格相较于常规编程风格,具有更高的可维护性、可重用性和可测试性。在本文中,我们将探讨 J...
    99+
    2023-11-05
    编程算法 函数 javascript
  • JavaScript函数式编程示例分析
    目录函数式编程函数柯理化(Curring)Compose场景案例总结函数式编程 1.函数式编程指的是函数的映射关系 2.vue3、react16.8的函数组件推动了前端函数编程 3....
    99+
    2022-11-13
    JavaScript函数式编程 JS函数式编程
  • JavaScript函数式编程实现介绍
    目录为什么要学习函数式编程什么是函数式编程前置知识函数是一等公民函数可以储存在变量中函数作为参数函数作为返回值高阶函数什么是高阶函数使用高阶函数的意义常用高阶函数闭包纯函数纯函数概念...
    99+
    2024-04-02
  • JavaScript中的函数式编程怎么应用
    本文小编为大家详细介绍“JavaScript中的函数式编程怎么应用”,内容详细,步骤清晰,细节处理妥当,希望这篇“JavaScript中的函数式编程怎么应用”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。JavaS...
    99+
    2023-06-27
  • es6箭头函数是不是函数式编程
    这篇文章主要讲解了“es6箭头函数是不是函数式编程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“es6箭头函数是不是函数式编程”吧!是。箭头函数是函数式编程的一种体现,函数式编程将更多的关注...
    99+
    2023-07-05
  • Python中的函数式编程的方法是什么
    这篇文章主要介绍“Python中的函数式编程的方法是什么”,在日常操作中,相信很多人在Python中的函数式编程的方法是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python中的函数式编程的方法是什么...
    99+
    2023-06-16
  • golang函数式编程的未来趋势是什么?
    函数式编程在 go 语言中正变得流行,它提供更简洁、更可预测的代码。函数式编程的核心概念是使用纯函数,这些函数给定相同的输入总是返回相同的结果,并且没有副作用。go 提供了一流函数、匿名...
    99+
    2024-04-30
    golang 函数式编程
  • golang函数式编程的最佳实践是什么?
    函数式编程在 go 语言中的最佳实践包括:避免可变状态,提高可预测性和并行潜力。使用不可变数据结构,防止意外修改并增强并发安全性。利用高阶函数,创建可重用和可组合的代码。使用惰性求值,优...
    99+
    2024-05-01
    golang 函数式编程
  • 什么是javascript函数
    这篇文章主要介绍“什么是javascript函数”,在日常操作中,相信很多人在什么是javascript函数问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”什么是javascript函数”的疑惑有所帮助!接下来...
    99+
    2023-06-14
  • JavaScript 函数是什么
    本篇文章为大家展示了JavaScript 之什么是函数,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。函数是一段代码,它只定义一次,但可...
    99+
    2024-04-02
  • JavaScript函数式编程的示例分析
    这篇文章给大家分享的是有关JavaScript函数式编程的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。看过许多关于函数式编程的讲解,但是其中大部分是停留在理论层面,还有...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作