• 函数式编程(Funtional Programming,简称FP)是一种编程范式,也就是如何编写程序的方法论
  1. 主要思想:把计算过程尽量分解成一系列可复用函数的调用
  2. 主要特征:函数是“第一等公民”:

    函数与其他数据类型一样的地位,可以赋值给其他变量,也可以作为函数参数、函数返回值
  • 函数式编程最早出现在LISP语言,绝大大部分的代码编程语言也对函数式编程做了不同程度的支持,比如:Haskell、JavaScript、Python、Swift、Kotlin、Scala等
  • 函数时编程中的几个常用的概念
  1. Higher-Order Function、Function Currying
  2. Functor、Applicatie Functor、Monad

FP实践 - 传统写法

  • 常规写法
// 假设要实现以下功能:[(num + 3) * 5 - 1] % 10 / 2
let num = 1
func add(_ v1: Int, _ v2: Int) -> Int { v1 + v2 }
func sub(_ v1: Int, _ v2: Int) -> Int { v1 - v2 }
func multiple(_ v1: Int, _ v2: Int) -> Int { v1 * v2 }
func divide(_ v1: Int, _ v2: Int) -> Int { v1 / v2 }
func mod(_ v1: Int, _ v2: Int) -> Int { v1 % v2 }
let result = divide(mod(sub(multiple(add(num, 3), 5), 1), 10), 2)
print(result) // 4

FP实践 - 函数式写法

  • 柯里化
func add(_ v: Int) -> (Int) -> Int { { $0 + v } }
func sub(_ v: Int) -> (Int) -> Int { { $0 - v } }
func multiple(_ v: Int) -> (Int) -> Int { { $0 * v } }
func divide(_ v: Int) -> (Int) -> Int { { $0 / v } }
func mod(_ v: Int) -> (Int) -> Int { { $0 % v } }
  • 函数合成
func composite(_ f1: @escaping (Int) -> Int,
_ f2: @escaping (Int) -> Int) -> (Int) -> Int {
return { f2(f1($0)) }
}
let fn = composite(add(3), multiple(5))
print(fn(num)) // 20
  • 函数合成 - 利用符号
infix operator >>>: AdditoinPrecedence
func >>>(_ f1: @escaping (Int) -> Int,
_ f2: @escaping (Int) -> Int) -> (Int) -> Int { return { f2(f1($0)) } }
let fn = add(3) >>> multiple(5)
print(fn(num)) // 20 函数合成 - 利用符号 - 泛型
infix operator >>>: AdditoinPrecedence
func >>> <A, B, C>(_ f1: @escaping (A) -> B,
_ f2: @escaping (B) -> C) -> (A) -> C { return { f2(f1($0)) } }
let fn = add(3) >>> multiple(5)
print(fn(num)) // 20
let fn = add(3) >>> multiple(5) >>> sub(1) >>> mod(10) >>> divide(2)
print(fn(num)) // 4

高阶函数(Higher-Order Function)

  • 高阶函数是至少满足下列一个条件的函数:
  1. 接受一个或多个函数作为输入(map、filter、reduce等)
  2. 返回一个函数
  • FP中到处都是高阶函数

柯里化(Currying)

  • 将一个接受多参数的函数变换为一系列只接受单个参数的函数
func add1(_ v1: Int, _ v2: Int) -> Int { v1 + v2 }
add1(10, 20)
func add1(_ v: Int) -> (Int) -> Int { { $0 + v } }
add1(10)(20)
  • Array、Optional的map方法接受的参数就是一个柯里化函数
  • 三个参数柯里化
func add2(_ v1: Int, _ v2: Int, v3: Int) -> Int { v1 + v2 + v3 }
func add2(_ v3: Int) -> (Int) -> (Int) -> Int {
// v2 == 20
return { v2 in
// v1 == 10
return { v1 in
return v1 + v2 + v3
}
}
}
add2(10, 20, 30)
add2(30)(20)(10)
  • 两个参数柯里化 - 泛型
func currying<A, B, C>(_ fn: @escaping (A, B) -> C) -> (B) -> (A) -> C {
return { b in
return { a in
return fn(a, b)
}
}
}
currying(add1(20)(10))
  • 两个参数柯里化- 利用符号 - 泛型
prefix func ~<A, B, C>(_ fn: @escaping (A, B) -> C)
-> (B) -> (A) -> C {
{ b in { a in fn(a, b) } }
}
print((~sub(20)(10)) // -10
// 合成函数-> 泛型柯里化
let fn = (~add)(3) >>> (~multiple)(5) >>> (~sub)(1) >>> (~mod)(10) >>> (~divide)(2)
print(fn(1)) // 4
  • 三个参数柯里化 - 利用符号 - 泛型
prefix func ~<A, B, C, D>(_ fn: @escaping (A, B, C) -> D)
-> (C) -> (B) -> (A) -> D {
{ c in { b in { a in fn(a, b, c) } } }
}
print((~add2)(30)(20)(10)) // 60

函子(Functor)

  • 像Array、Optional这样支持map运算的类型,成为函子(Functor)
  • 怎么样的Type才能称之为函子呢?
func map<T>(_ fn: (Inner) -> T) -> Type<T>
  • 数组和可选项是符合函子规定的公式:
func map<T>(_ fn: (Element) -> T) -> Array<T>   // [T]
func map<T>(_ fn: (Wrapped) -> T) -> Opional<T> // T?

适用函子(Applicative Functor)

  • 对任意一个函子F,如果能支持一下运算,该函子就是一个适用函子
func pure<A>(_ value: A) -> F<A>
func <*><A, B>(fn: F<(A) -> B>, value: F<A>) -> F<B>
  • Optional可以成为适用函子
func pure<A>(_ value: A) -> A? { value }
infix operator <*>: AdditionPrecedence
func <*><A, B>(fn: ((A) -> B)?, value: A?) -> B? {
guard let f = fn, let v = value else { return nil }
return f(v)
}
var value: Int? = 10
var fn: ((Int) -> Int)? = { $0 * 2 }
// Optional
print(fn <*> value as Any)

![在这里插入图片描述](https://img-blog.csdnimg.cn/20190923152402722.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3Nvbmd6aHVvMTk5MQ==,size_16,color_FFFFFF,t_70 =240x300)

  • Array可以成为适用函子
func pure<A>(_ value: A) -> [A] { [value] }
func <*><A, B>(fn: [(A) -> B], value: [A]) -> [B] {
var arr: [B] = []
if fn.count == value.count {
for i in fn.startIndex..<fn.endIndex {
arr.append(fn[i](value[i]))
}
}
return arr
}
// [10]
print(pure(10))
var arr = [{ $0 * 2 }, { $0 + 10 }, { $0 - 5 }] <*> [1, 2, 3]
// [2, 12, -2]
print(arr)

单子(Monad)

  • 对于任意一个类型F,如果能支持一下运算,那么就可以成为是一个单子(Monad)
func pure<A>(_ value: A) -> F<A>
func flatMap<A, B>(_ value: F<A>, _ fn: (A) -> F<B>) -> F<B>
  • 很显然,Array、Optional都是单子

编程范式 --- 函数式编程(Funtional Programming,简称FP)的更多相关文章

  1. Scala编程入门---函数式编程之集合操作

    集合的函数式编程: 实战常用: //map案例实战:为List中的每个元素都添加一个前缀. List("leo","Jen","peter" ...

  2. Scala编程入门---函数式编程

    高阶函数 Scala中,由于函数时一等公民,因此可以直接将某个函数传入其他函数,作为参数.这个功能是极其强大的,也是Java这种面向对象的编程语言所不具备的. 接收其他函数作为函数参数的函数,也被称作 ...

  3. Scala函数与函数式编程

    函数是scala的重要组成部分, 本文将探讨scala中函数的应用. scala作为支持函数式编程的语言, scala可以将函数作为对象即所谓"函数是一等公民". 函数定义 sca ...

  4. swift之函数式编程

    函数式编程初探 最近初学swift,和OC比,发现语言更现代,也有了更多的特性.如何写好swift代码,也许,熟练使用新特性写出更优秀的代码,就是答案.今天先从大的方向谈谈swift中的编程范式-函数 ...

  5. Python 进阶(一)函数式编程

    来自慕课网: 简介: 函数:function ,在入门课程已学 函数式:functional,一种编程范式 函数式编程是一种抽象计算的编程模式,函数≠函数式,好比:计算≠计算机

  6. javascript 函数式编程

    编程范式 编程范式是一个由思考问题以及实现问题愿景的工具组成的框架.很多现代语言都是聚范式(或者说多重范式): 他们支持很多不同的编程范式,比如面向对象,元程序设计,泛函,面向过程,等等. 函数式编程 ...

  7. Python进阶之函数式编程(把函数作为参数)

    什么是函数式编程? 什么是函数式编程? 函数:function 函数式:functional,一种编程范式 函数式编程是一种抽象计算的编程模式 函数≠函数式,比如:计算≠计算机 在计算机当中,计算机硬 ...

  8. 编程范式:命令式编程(Imperative)、声明式编程(Declarative)和函数式编程(Functional)

    主要的编程范式有三种:命令式编程,声明式编程和函数式编程. 命令式编程: 命令式编程的主要思想是关注计算机执行的步骤,即一步一步告诉计算机先做什么再做什么. 比如:如果你想在一个数字集合 collec ...

  9. Lambda01 编程范式、lambda表达式与匿名内部类、函数式接口、lambda表达式的写法

    1 编程范式 主要的编程范式有三种:命令式编程,声明式编程和函数式编程. 1.1 命令式编程 关注计算机执行的步骤,就是告诉计算机先做什么后做什么 1.2 声明式编程 表达程序的执行逻辑,就是告诉计算 ...

随机推荐

  1. 本地项目上传到github上最直接步骤

    1.首先得有一个git账号(本地安装git) 2.git上创建一个project 3.回到本地你要提交文件夹位置 4.按住shift + 鼠标右键 选择在此处打开命令窗口 5.输入命令  git in ...

  2. Redis学习总结(五)--Redis集群创建

    在之前我们讲到了主从,但是对于大数据量的场景下我们就需要用到集群了,让我们来了解下集群吧. 为什么需要集群 单机内存太小 redis最高可以达到10万/s 请求,如果超过该频率呢? 数据分布方式 数据 ...

  3. python学习——高阶函数

    递归函数 在函数内部,可以调用其他函数.如果一个函数在内部调用自身本身,这个函数就是递归函数.使用递归函数的优点是逻辑简单清晰,缺点就是过深的调用会导致栈溢出.但是针对尾递归优化的语言可以通过尾递归防 ...

  4. pyinstaller打包出错numpy.core.multiarray failed to import

    py原文件运行时正常,但用pyinstaller打包为exe后,在运行则报错: 这是因为cv2要求的numpy版本与你装的numpy版本不一样,导致冲突:网上很多说升级numpy,但你把numpy升的 ...

  5. 《GO Home Trash!》UML类图,ER图以及数据库设计

    <Go Home Trash!>UML类图 ER图以及数据库中数据表 分析: 这款软件经过我们前期的讨论以及需求分析,确定了用户,客服以及管理员三个实体.在设计UML类图时,对各个实体之间 ...

  6. 洛谷P2331 [SCOI2005]最大子矩阵 DP

    P2331 [SCOI2005]最大子矩阵 题意 : 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. 第一行为n,m,k(1≤n≤ ...

  7. WPF 字体图标样式

    1.在网上下载Font ICO 字体字体压缩包 Font Awesome 矢量图标库    2.下载到的压缩包,解压后获得图标字体文件        3.然后把字体文件夹拷贝到项目中,目录为(/fon ...

  8. 【LeetCode】17-电话号码的字母组合

    题目描述 给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合. 给出数字到字母的映射如下(与电话按键相同).注意 1 不对应任何字母. 示例: 输入:"23" 输出: ...

  9. 开发必配的Finder设置

    1.显示标签页.显示路径栏.显示状态栏的设置位置,在访达->显示-> 显示状态栏 个人三个都设置了,但是觉得显示状态栏用的并不多,反而多一行,下面是显示状态栏的效果,主要可以一眼看出有多少 ...

  10. Helm 从入门到实践 | 从 0 开始制作一个 Helm Charts

    本周 Helm 官方发布博客,指导用户从 v2 迁移到 v3,这标志 Helm 逐渐走向成熟.早在今年 6 月,阿里云就正式发布了国内首个 Helm Hub 中国镜像站:开放云原生应用中心 - Clo ...