函数式编程是种编程方式,它将电脑运算视为函数的计算。函数编程语言最重要的基础是λ演算(lambda calculus),而且λ演算的函数可以接受函数当作输入(参数)和输出(返回值)。和指令式编程相比,函数式编程强调函数的计算比指令的执行重要。和过程化编程相比,函数式编程里函数的计算可随时调用。

  最近一直在研究函数式编程,从函数式编程中仿佛看到了js规范化的影子。大家都知道js是一门很灵活的编程语言,而这种灵活性在代码量的累积下会产生质量和可控性的问题。初学js的朋友大多都是结果导向,写的代码刚刚够用现有的项目,很难修改与扩展代码,而且不规范的代码常常让队友甚至是自己头疼。而函数式编程却能改变这样的现状。

  再说函数式编程之前按照惯例要说一些相关的知识。

1.什么是纯函数?

  纯函数顾名思义就是很纯的函数,大家都喜欢纯粹的事物,那什么样的函数才配称作纯函数呢?
  简单地说,纯函数就是传入相同的参数可以返回相同的值的函数,也就是说返回的结果只依赖于传递的参数,而和环境没有关系。类似于我们数学中的函数的概念。

Array.prototype.map()
Array.prototype.slice()
String.prototype.toUpperCase()

  像上面这样的函数就是纯函数

Array.prototype.sort()
Math.random()

  像上面这样的函数就不是纯函数了
  纯函数有很多优秀的性质让我们喜欢,

1.纯洁性

  她既不会被环境改变也不会改变环境,结果只依赖传入值。那么这就方便我们对函数进行缓存。下面以求x的平方为例(这里用到了ramda函数库,没接触也没关系,这里只是讲概念,之后我们会提及这个函数库)

let square = R.memoizeWith(R.identity, x => x * x);
square(3); // => 9 第一次会调用原始方法并将参数和结果以key-value的形式存储。
square(3); // => 9 这里是调用缓存
square(3); // => 9 这里也是调用缓存

2.易测试性

  由于纯函数执行不需要环境,所以进行测试时我们只需要面对函数本身测试就可以了。这让函数测试变得非常简单

3.透明性

  如果一段代码可以替换成它执行所得的结果,而且是在不改变整个程序行为的前提下替换的,那么我们就说这段代码是引用透明的。由于纯函数总是能够根据相同的输入返回相同的输出,所以它们就能够保证总是返回同一个结果,这也就保证了引用透明性。

2.什么是柯里化?

  函数柯里化(curry)的定义很简单:传递给函数一部分参数来调用它,让它返回一个函数去处理剩下的参数。下面我们对比一下柯里化之前的函数和柯里化的函数。

let add = (x, y, z) => x + y + z;
add(1,2,3) // => 6;
let add = (x, y, z) => {
return z => {
return y => {
return x + y + z;
}
}
}
add(1)(2)(3) // => 6;

  后者就是柯里化的函数,它利用了函数的记忆性记住了每次传你进去的参数并返回了可继续执行的函数。是不是感觉发现了新大陆?

3.什么是函数组合

  函数组合(compose)就是把多个纯函数组合起来解决函数嵌套问题。下面我们来看两个例子

let compose = (f, g, h) => x => f(g(h(x)));
let add = x => x + 4;
let mul = x => x * 2;
let mis = x => x - 1;
compose(add, mul, mis)(1) // => 3
let compose = (f, g, h) => x => f(g(h(x)));
let getFirst = arr => arr[0];
let getUpperCase = str => str.toUpperCase();
let getReverse = arr => arr.reverse();
compose(getUpperCase, getFirst, getReverse)(['v', 'a', 'd', 'i', 'm']); // => 'M'

  因为纯函数的纯洁性,我们可以把多个函数组合起来,注意一定是纯函数。

4.什么是point-free

point-free简单地说就是无需使用所要处理的值,只关注运算过程。可以用公式 fn = R.pipe(f1, f2, f3);表示,也就是说如果事先定义了函数f1,f2,f3就可以推算出函数fn,因此无需使用参数形式,下面我们来对比一下两种形式。

let noPointFree = word => word.toUpperCase().split('-');
let pointFree = compose(split('-'),toUpperCase);

  pointfree 模式能够帮助我们减少不必要(中间变量)的命名,让代码保持简洁和通用。

  关于函数式编程先讲一下基础的知识,下一节我来讲一下js函数式编程优秀的函数库----Ramda函数库。

原创博客:转载请注明从函数式编程到Ramda函数库(一)

从函数式编程到Ramda函数库(一)的更多相关文章

  1. 从函数式编程到Ramda函数库(二)

    Ramda 基本的数据结构都是原生 JavaScript 对象,我们常用的集合是 JavaScript 的数组.Ramda 还保留了许多其他原生 JavaScript 特性,例如,函数是具有属性的对象 ...

  2. 【Python】[函数式编程]高阶函数,返回函数,装饰器,偏函数

    函数式编程高阶函数 就是把函数作为参数的函数,这种抽象的编程方式就是函数式编程.--- - -跳过,不是很理解,汗 - ---

  3. Python之路(第七篇)Python作用域、匿名函数、函数式编程、map函数、filter函数、reduce函数

    一.作用域 return 可以返回任意值例子 def test1(): print("test1") def test(): print("test") ret ...

  4. python函数式编程之返回函数、匿名函数、装饰器、偏函数学习

    python函数式编程之返回函数 高阶函数处理可以接受函数作为参数外,还可以把函数作为结果值返回. 函数作为返回值 def laxy_sum(*args): def sum(): ax = 0; fo ...

  5. Python之路Python作用域、匿名函数、函数式编程、map函数、filter函数、reduce函数

    Python之路Python作用域.匿名函数.函数式编程.map函数.filter函数.reduce函数 一.作用域 return 可以返回任意值例子 def test1(): print(" ...

  6. Learning Python 012 函数式编程 2 返回函数 匿名函数 装饰器 偏函数

    Python 函数式编程 2 返回函数 返回函数的意思就是:函数作为返回值.(高阶函数除了可以接受函数作为参数外,还可以把函数作为结果值返回.) 举个例子:实现一个可变参数的求和. 正常的函数: de ...

  7. 跟着ALEX 学python day3集合 文件操作 函数和函数式编程 内置函数

    声明 : 文档内容学习于 http://www.cnblogs.com/xiaozhiqi/  一. 集合 集合是一个无序的,不重复的数据组合,主要作用如下 1.去重 把一个列表变成集合 ,就自动去重 ...

  8. 理解函数式编程中的函数组合--Monoids(二)

    使用函数式语言来建立领域模型--类型组合 理解函数式编程语言中的组合--前言(一) 理解函数式编程中的函数组合--Monoids(二) 继上篇文章引出<范畴论>之后,我准备通过几篇文章,来 ...

  9. JavaScript函数式编程(纯函数、柯里化以及组合函数)

    JavaScript函数式编程(纯函数.柯里化以及组合函数) 前言 函数式编程(Functional Programming),又称为泛函编程,是一种编程范式.早在很久以前就提出了函数式编程这个概念了 ...

随机推荐

  1. Confluence 6 如何备份和恢复

    如何备份 有关备份你数据库使用的命令与你使用的具体数据库有关,例如针对 PostgreSQL 数据库,备份命令为 pg_dump dbname > outfile. 你应该针对你使用的数据库不同 ...

  2. SpringBoot全局日志管理(AOP)

    1.在pom.xml中引入aop的jar包 <dependency> <groupId>org.springframework.boot</groupId> < ...

  3. Matplotlib模块:绘图和可视化

    一.简单介绍Matplotlib 1.Matplotlib是一个强大的Python绘图和数据可视化的工具包 2.安装方法:pip install matplotlib 3.引用方法:import ma ...

  4. django rest framework(4)

    目录 一.分页 二.视图 三.路由 四.渲染器 一.分页 试问如果当数据量特别大的时候,你是怎么解决分页的? 方式a.记录当前访问页数的数据id 方式b.最多显示120页等 方式c.只显示上一页,下一 ...

  5. ubuntu 安装Mysql8.0

    1. 去官网下载安装包 下载链接:点击打开链接 https://dev.mysql.com/downloads/mysql/ 如果你的系统是32位选择第一个,64位选择第二个 也可以用wget 下载 ...

  6. bootstrap和easyui

    1.easyui:自定义的样式要在原先的easyui样式之前引入,这样自定义的样式才能把原先的样式覆盖,即放置顺序为: <link rel="stylesheet" href ...

  7. Oracle logminer 日志挖掘

    Table of Contents 1. LOGMNR简介 2. 创建数据字典 2.1. 外部文件存储数据字典 2.2. redo log 存储数据字典 3. 添加需要分析的文件 4. 开始分析文件 ...

  8. 微信JSAPI分享朋友圈调试经验:invalid signature签名错误排查

    .invalid signature签名错误.建议按如下顺序检查: 1.确认签名算法正确,可用http://mp.weixin.qq.com/debug/cgi-bin/sandbox?t=jsapi ...

  9. 升级到 Android Studio 3.0 + Gradle 4.1 遇到的一些坑及解决方案

    问题一: Cannot set the value of read-only property 'outputFile' for ApkVariantOutputImpl_Decorated{apkD ...

  10. Select2 多层次赋值时异步赋值的问题

    场景: 当选择人员时加载人员,选择部门时加载部门.所以在人员下,选择人员A后,如果选择部门,会触发二级select 重新获取数据. 问题: 使用select2()方法进行绑定远程数据后,对第二个sel ...