找回密码
 立即注册
注册 登录
×
热搜: 活动 交友 discuz
查看: 80|回复: 6

什么是「函数式编程」?

[复制链接]

2

主题

31

帖子

61

积分

注册会员

Rank: 2

积分
61
发表于 2023-7-30 12:07:40 | 显示全部楼层 |阅读模式
上一篇文章我们讲解了闭包的机制 从编译原理角度认识 javascript中的「闭包」,提到闭包可以理解为 定义在一个函数内部的函数,将内部函数作为返回值。这正体现了函数是一等公民的特点。而这正是我们要说的函数式编程的两个基本特征之。
到底什么是函数式编程?

其实,函数式编程是一种编程范式,除了函数式编程之外还有 命令式编程,声明式编程 等编程范式。
命令式编程
命令式编程 是面向计算机硬件的抽象,有变量、赋值语句、表达式、控制语句等,可以理解为 命令式编程就是冯诺伊曼的指令序列。 它的主要思想是关注计算机执行的步骤,即一步一步告诉计算机先做什么再做什么。
比如,我们要查找数组 numList 中大于5的所有数字,需要这样告诉计算机:

  • 创建一个存储结果的集合变量 results
  • 遍历这个数字集合 numList;
  • 一个一个地判断每个数字是不是大于 5,如果是就将这个数字添加到结果集合变量 results 中。
let results = [];
for(let i = 0; i < numList.length; i++){
    if(numList > 5){
        results.push(numList)
    }
}
声明式编程
声明式编程 是以数据结构的形式来表达程序执行的逻辑。它的主要思想是告诉计算机应该做什么,但不指定具体要怎么做。SQL 语句就是最明显的一种声明式编程的例子,例如:
SELECT * FROM collection WHERE num > 5除了 SQL,网页编程中用到的 HTML 和 CSS 也都属于声明式编程。它的特点:

  • 它不需要创建变量用来存储数据
  • 另一个特点是它不包含循环控制的代码如 for, while
函数式编程
而函数式编程和声明式编程是有所关联的,因为他们思想是一致的:即只关注做什么而不是怎么做。但函数式编程不仅仅局限于声明式编程
函数式编程是面向数学的抽象,将计算描述为一种表达式求值,其实,函数式程序就是一个表达式。
函数式编程本质
函数式编程中函数并部署指计算机中的函数,而是指数学中的函数,即自变量的映射。函数的值取决于函数的参数的值,不依赖于其他状态,比如abs(x)函数计算x的绝对值,只要x不变,无论何时调用、调用次数,最终的值都是一样。
函数式编程的特点

  • 函数是第一等公民
  • 函数是纯函数
接下来我们分别介绍下函数式编程的这两个特点
函数是第一等公民
函数是第一等公民:是指函数跟其它的数据类型一样处于平等地位,可以赋值给其他变量,可以作为参数传入另一个函数,也可以作为别的函数的返回值。正如我们开头提到的闭包的实现好体现了这个特点,例如如下代码:
// 赋值
var func1 = function func1() {  }
// 函数作为参数
function func2(fn) {
    fn()
}   
// 函数作为返回值
function func3() {
    return function() {}
}
函数是纯函数

纯函数是指相同的输入总会得到相同的输出,并且不会产生副作用的函数。纯函数的两个特点:

  • 相同的输入必有同输出
  • 没有副作用
无副作用 指的是函数内部的操作不会对外部产生影响(如修改全局变量的值、修改 dom 节点等)。
// 是纯函数
function sum(x,y){
    return x + y
}
// 输出不确定,不是纯函数
function random(x){
    return Math.random() * x
}
// 有副作用,不是纯函数
function setFontSize(el,fontsize){
    el.style.fontsize = fontsize ;
}
// 输出不确定、有副作用,不是纯函数
let count = 0;
function addCount(x){
    count+=x;
    return count;
}
函数式编程的基本运算
函数合成(compose)
指的是将代表各个动作的多个函数合并成一个函数。
上面讲到,函数式编程是对过程的抽象,关注的是动作。看下下面的例子
function add(x) {
    return x + 10
}
function multiply(x) {
    return x * 10
}

console.log(multiply(add(2)))  // 120
将合成的动作抽象为一个函数 compose如下:
function compose(f,g) {
    return function(x) {
        return f(g(x));
    };
}
// 这样我们我们可以通过如下的方式得到合成函数
// 执行动作的顺序是从右往左
let calculate=compose(multiply,add);
console.log(calculate(2))  // 120
只要往 compose 函数中传入代表各个动作的函数,我们便能得到最终的合成函数。但上述 compose 函数的局限性是只能够合成两个函数,如果需要合成的函数不止两个呢,所以需要一个通用的 compose 函数。
function compose() {
  let args = arguments;
  let start = args.length - 1;
  return function () {
    let i = start - 1;
    let result = args[start].apply(this, arguments);
    while (i >= 0){
      result = args.call(this, result);
      i--;
    }
    return result;
  };
}

// 使用
function add(str){
    return x + 10
}
function multiply(str) {
    return x * 10
}
function minus(str) {
    return x - 10
}

let composeFun = compose(minus, multiply, add);
composeFun(2) // 110
通过 compose 将上述三个动作代表的函数合并成了一个,并最终输出了正确的结果。
函数柯里化(Currying)
函数柯里化又称部分求值。一个柯里化的函数首先会接受一些参数,接受了这些参数之后,该函数并不会立即求值,而是继续返回另外一个函数,刚才传入的参数在函数形成的闭包中被保存起来。待到函数被真正需要求值的时候,之前传入的所有参数都会被一次性用于求值
柯里化函数有如下两个特性:

  • 接受一个单一参数
  • 返回接受余下的参数而且返回结果的新函数
function sum(a, b) {
    return a + b;
}
console.log(sum(2, 2)) // 4
假设函数 sum 的柯里化函数是 sumCurry,那么从上述定义可知,sumCurry(2)(2) 应该实现与上述代码相同的效果,输出 4 。这里我们可以比较容易的知道,sumCurry 的代码如下
// sumCurry 是 sum 的柯里化函数
function sumCurry(a) {
    return function(b) {
        return a + b;
    }
}
console.log(sumCurry(2)(2));  // 4
如果有一个函数 createCurry 能够实现柯里化,那么我们便可以通过下述的方式来得出相同的结果
// sumCurry 返回一个柯里化函数
var sumCurry=createCurry(sum);
console.log(sumCurry(2)(2));  // 4
可以看到,函数 createCurry 传入一个函数 sum 作为参数,返回了一个柯里化函数 sumCurry,函数 sumCurry 能够处理 sum 中的剩余参数。这个过程就称为函数柯里化,我们称 sumCurry 是 add 的柯里化函数。
怎么得到实现柯里化的函数 createCurry 呢?这里我直接给出 createCurry 的代码
// 参数只能从左到右传递
function createCurry(func, arrArgs) {
    var args=arguments;
    var funcLength = func.length;
    var arrArgs = arrArgs || [];

    return function() {
        var _arrArgs = Array.prototype.slice.call(arguments);
        var allArrArgs=arrArgs.concat(_arrArgs)

        // 如果参数个数小于最初的func.length,则递归调用,继续收集参数
        if (allArrArgs.length < funcLength) {
            return args.callee.call(this, func, allArrArgs);
        }

        // 参数收集完毕,则执行func
        return func.apply(this, allArrArgs);
    }
}

// createCurry 返回一个柯里化函数
var sumCurry=createCurry(function(a, b, c) {
    return a + b + c;
});
sumCurry(1)(2)(3) // 6
sumCurry(1, 2, 3) // 6
sumCurry(1)(2,3) // 6
sumCurry(1,2)(3) // 6
柯里化实际上是把简答的问题复杂化了,但是复杂化的同时在使用函数时拥有了更加多的自由度。
柯里化用途
现在需要实现一个功能,将一个全是数字的数组中的数字转换成百分数的形式。按照正常的逻辑,我们可以按如下代码实现
function getPercentList(array) {
    return array.map(function(item) {
        return item * 100 + '%'
    })
}

console.log(getPercentList([1, 0.2, 3, 0.4]));   
// 结果:['100%', '20%', '300%', '40%']
如果通过柯里化的方式来实现
function map(func, array) {
    return array.map(func);
}
var mapCurry = createCurry(map);
var getNewArray = mapCurry(function(item) {
    return item * 100 + '%'
})
console.log(getPercentList([1, 0.2, 3, 0.4]));
// 结果:['100%', '20%', '300%', '40%']
上述例子太简单以致不能表现出柯里化的强大,具体柯里化的使用还需要结合具体的场景,其实,没有必要为了柯里化而柯里化,不管用什么方式我们的最终目的都是为了更好地解决问题。
高阶函数
满足下列条件之一的函数就可以称为高阶函数:

  • 函数作为参数被传递
把函数当作参数传递,这代表我们可以抽离出一部分容易变化的业务逻辑,把这部分业务逻辑放在函数参数中,这样一来可以分离业务代码中变化与不变的部分。其中一个重要应用场景就是常见的回调函数。
下面例子中js的函数都是对高阶函数的利用:
[1, 4, 2, 5, 0].sort((a, b) => a - b);
// [0, 1, 2, 4, 5]
        
[0, 1, 2, 3, 4].map(v => v + 1);
// [1, 2, 3, 4, 5]
        
[0, 1, 2, 3, 4].every(v => v < 5);
// true
2.函数作为返回值输出
让函数继续返回一个可执行的函数,意味着运算过程是可延续的
const fn = (() => {
    let students = [];
    return {
        addStudent(name) {
            if (students.includes(name)) {
                return false;
            }
            students.push(name);
        },
        showStudent(name) {
            if (Object.is(students.length, 0)) {
                return false;
            }
            return students.join(",");
        }
    }
})();
fn.addStudent("liming");
fn.addStudent("zhangsan");
fn.showStudent(); //输出:liming,zhangsan
同时满足两个条件的高阶函数
const plus = (...args) => {
    let n = 0;
    for (let i = 0; i < args.length; i++) {
        n += args;
    }
    return n;
}

const mult = (...args) => {
    let n = 1;
    for (let i = 0; i < args.length; i++) {
        n *= args;
    }
    return n;
}

const createFn = (fn) => {
    let obj = {};
    return (...args) => {
        let keyName = args.join("");
        if (keyName in obj) {
            return obj[keyName];
        }
        obj[keyName] = fn.apply(null, args);
        return obj[keyName];
    }
}

let fun1 = createFn(plus);
console.log(fun1(2, 2, 2)); //输出:6

let fun2 = createFn(mult);
console.log(fun2(2, 2, 2)); //输出:8
参考:函数式编程
回复

使用道具 举报

2

主题

7

帖子

6

积分

新手上路

Rank: 1

积分
6
发表于 2023-7-30 12:08:00 | 显示全部楼层
createCurry 是不是写错了 return args.callee.call(this, func, allArrArgs);
args 函数上下文没出现过
回复

使用道具 举报

1

主题

3

帖子

4

积分

新手上路

Rank: 1

积分
4
发表于 2023-7-30 12:08:28 | 显示全部楼层
我看看
回复

使用道具 举报

4

主题

49

帖子

96

积分

注册会员

Rank: 2

积分
96
发表于 2023-7-30 12:08:40 | 显示全部楼层
“函数式编程中的函数并部署指计算机中的函数,而是指数学中的函数,即自变量的映射。“
“并部署指“意思是“并不是”
回复

使用道具 举报

4

主题

9

帖子

15

积分

新手上路

Rank: 1

积分
15
发表于 2023-7-30 12:09:17 | 显示全部楼层
“命令式编程”是不是又叫“过程式编程”呀?
回复

使用道具 举报

0

主题

38

帖子

71

积分

注册会员

Rank: 2

积分
71
发表于 2023-7-30 12:09:58 | 显示全部楼层
面向过程
回复

使用道具 举报

1

主题

45

帖子

86

积分

注册会员

Rank: 2

积分
86
发表于 2023-8-28 13:17:38 | 显示全部楼层
OMG!介是啥东东!!!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

Archiver|手机版|小黑屋| 问天社区-最全面的网络问答中心

GMT+8, 2025-3-17 00:49 , Processed in 0.117645 second(s), 20 queries .

Powered by Discuz! X3.4

Copyright © 2020, LianLian.

快速回复 返回顶部 返回列表