二十二、函数式编程、面向协议式编程

函数式编程(Functional Prigramming)

函数式编程

  1. 函数式编程(Functional Prigramming,简称FP)是一种编程范式,也就是如何编写程序的方法论
  • 主要思想:把计算过程尽量分解成一系列可服用函数的调用
  • 主要特征:函数是"第一等公民"
    ✅函数与其他数据类型一样的地位,可以赋值给变量,也可以作为函数参数、函数返回值
  1. 函数式编程最早出现在LISP语言,绝大部分语言也对函数式编程做了不同程度的支持

  2. 函数式编程中几个常用的概念

  • Higher-Order Function、Function Currying
  • Functor、Applicative Functor、Monad
  1. 参考资料:
    函数式编程
    函数式编程
FP实践

假如要实现以下功能: [(num + 3) * 5 - 1] % 10 / 2 这个过程用函数实现应该怎么做?
☝️第一种:传统写法

var 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
}

//不仅嵌套多层 且难以理解
divide(mod(sub(multiple(add(num, 3), 5), 1), 10), 2)

✌️第二种写法:

//函数式写法
func fp_add(_ v:Int) -> ((Int) -> Int){
    {
        $0 + v
    }
}

func fp_sub(_ v:Int) -> ((Int) -> Int){
    {
        $0 - v
    }
}

func fp_multiple(_ v:Int) -> ((Int) -> Int){
    {
        $0 * v
    }
}

func fp_divide(_ v:Int) -> ((Int) -> Int){
    {
        $0 / v
    }
}

func fp_mod(_ v:Int) -> ((Int) -> Int){
    {
        $0 % v
    }
}

infix operator >>> : AdditionPrecedence
func >>><A, B, C>(_ f1:@escaping (A) -> B,_ f2:@escaping (B) -> C) -> (A) -> C {
    {f2(f1($0))}
}

var fn = fp_add(3)>>>fp_multiple(5)>>>fp_sub(1)>>>fp_mod(1)>>>fp_mod(2)
fn(num)
高阶函数
  1. 高阶函数至少满足下列一个条件的函数
  • 接受一个或多个函数作为输入(map,filter,reduce等)
  • 返回一个函数
  1. FB到处是高阶函数
柯里化(Currying)
  • 将一个接受多个参数的函数变换为一系列只接受单个参数的函数
func add(_ v1: Int, _ v2: Int) -> Int { v1 + v2 }
//调用 add(10,20) 顺序:10 + 20

func add(_ v: Int) -> (Int) -> Int { { $0 + v } }
//调用 add(20)(10) 顺序:先创建一个 X + 20 的函数 然后传10

Tip:Array、Optional的map方法接收的参数就是一个柯里化函数

三个数相加函数的Curring如下:

//传统的
func add2(_ v1: Int, _ v2: Int, _ v3: Int) -> Int { v1 + v2 + v3 }

//柯⾥化
//v3 == 30
func add2(_ v3: Int) -> (Int) -> (Int) -> (Int) {
    //v2 == 20
    return { v2 in
        //v1 == 10
        return { v1 in
            return v3 + v2 + v1
        }
    }
}
add2(30)(20)(10) //10 + 20 + 30 = 60
  • 将函数柯里化
//方法1:定义函数将原函数传入,内部柯里化
func add1(_ v1: Int, _ v2: Int) -> Int { v1 + v2 }
func add2(_ v1: Int, _ v2: Int, _ v3: Int) -> Int { v1 + v2 + v3 }

func currying<A, B, C>(_ fn: @escaping (A, B) -> C) -> (B) -> (A) -> C {
    { b in { a in fn(a, b) } }
}
func currying<A, B, C, D>(_ fn: @escaping (A, B, C) -> D) -> (C) -> (B) -> (A) -> D {
    { c in { b in { a in fn(a, b, c) } } }
}

//传入一个函数,将函数柯里化
currying(add1)(20)(10) //add1(10,20) 10 + 20 = 30  20传给b,10传给a
currying(add2)(30)(20)(10) //add2(10,20,30) 10 + 20 + 30 = 60  30传给c,20传给b,10传给a
//自定义运算符
func add(_ v1: Int, _ v2: Int, v3: Int) -> Int{
    v1 + v2 + v3
}

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) } } }
}

(~add)(30)(20)(10)

再次回顾上面的例子(加减乘除):

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 }

//重载~运算符,将函数柯⾥化
prefix func ~<A, B, C>(_ fn: @escaping (A, B) -> C) -> (B) -> (A) -> C {
    { b in { a in fn(a, b) } }
}

//⾃定义>>>运算符
infix operator >>> : AdditionPrecedence
func >>><A, B, C>(_ f1: @escaping (A) -> B,
                  _ f2: @escaping (B) -> C) -> (A) -> C {
    { f2(f1($0)) }
}

var num = 1
var fn = (~add)(3) >>> (~multiple)(5) >>> (~sub)(1) >>> (~mod)(10) >>> (~divide)(2)
fn(num) //[(num + 3) * 5 - 1] % 10 / 2
函子(Functor)

我们先看下面这个函数:

func map<T>(_ fn: (Inner) -> T) -> Type<T>

支持如上map运算的类型才能称为函子:

  1. map运算要支持泛型
  2. 要求接收一个函数,这个函数把Type内部存放的数据当作参数传进去,返回一个T
  3. 返回的也是同一种Type<T>类型

因此Array、Optional也支持如上的运算即为函子

// Array<Element>
public func map<T>(_ transform: (Element) -> T) -> Array<T>
// Optional<Wrapped>
public func map<U>(_ transform: (Wrapped) -> U) -> Optional<U>

如何去理解函子:

如上该函子,函子里面包装的是2,对这个函子做+3的操作
函子解包取出里面的2,再将2做+3的操作,得到5,最后再将5又放到盒子里面,形成一个新的函子

如果这个函子是可选类型,那么map就不会调用,那么+3操作就不会执行,如下图

Optional类型

如果是数组,里面存放的是2 4 6,先将2 4 6取出来分别做相应的操作,最后操作的结果再包装成数组

Array类型
适用函子

对任意一个函子F,如果能支持以下运算,该函子就是一个适用函子

func pure<A>(_ value: A) -> F<A> //可以理解为,随便给⼀个值就能返回⾃⼰类型的泛型
func <*><A, B>(fn: F<(A) -> B>, value: F<A>) -> F<B> //可以理解为,给⼀个泛型F<A>和⼀个泛型函数fn,最后返回⼀个泛型B
  1. 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}
print(fn <*> value as Any) //Optional(20)

如何去理解适用函子:


将需要计算的操作也进行包装

适用函子
  1. Array是适用函子
func pure<A>(_ value: A) -> [A] { [value] }

infix operator <*> : AdditionPrecedence

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
}

//调用
print(pure(10)) // [10]

var arr = [{ $0 * 2}, { $0 + 10 }, { $0 - 5 }] <*> [1, 2, 3]
print(arr) // [2, 12, -2]
单子(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都是单子

面向协议式编程(Protocol Oriented Programming)

面向协议编程
  • 面向协议编程(Protocol Oriented Programming,简称POP),是Swift的一种编程范式
  • Swift也是一门面向对象的编程语言(Object Oriented Programming,简称OOP
  • 在Swift开发中,OOP和POP是相辅相成的,任何一方并不能取代另一方
回顾OOP
  • OOP的三大特性:封装、继承、多态

继承的经典使用场合:
当多个类(比如A、B、C类)具有很多共性时,可以将这些共性抽取到一个父类中(比如D类),最后A、B、C类继承D类

继承

但是OOP也存在一些不足:如何将 BVC、DVC 的公共方法 run 抽取出来?

如何基于OOP的解决方案?

解决方案:

  1. 将run方法放到另一个对象A中,然后BVC、DVC拥有对象A属性
  • 缺点:多了一些额外的依赖关系
  1. 将run方法增加到UIViewController分类中
  • 缺点:UIViewController会越来越臃肿,而且会影响它的其他所有子类
  1. 将run方法抽取到新的父类,采用多继承?(OC无法做到,C++支持多继承)
  • 缺点:会增加程序设计复杂度,产生菱形继承等问题,需要开发者额外解决
POP解决方案
protocol Runnable {
    func run()
}

extension Runnable {
    func run() {
        print("run")
    }
}

class BVC: UIViewController, Runnable {}
class DVC: UITableViewController, Runnable {}
POP的注意点
  • 优先考虑创建协议,而不是父类(基类)
  • 优先考虑值类型(struct、enum),而不是引用类型(class)
  • 巧用协议的扩展功能
  • 不要为了面向协议而使用协议
利用协议实现前缀效果
  • 值类型
//创建结构体
struct ZQ<Base> {
    let base: Base //Base是传⼊的类型
    init(_ base: Base) { //base是传⼊的类型的值
        self.base = base
    }
}

//创建协议,并给协议扩展类型和实例计算属性
protocol ZQCompatible {}
extension ZQCompatible {
    static var ll: ZQ<Self>.Type { //获取ZQ<Base>类型属性
        get { ZQ<Self>.self }
        set {}
    }
    
    var ll: ZQ<Self> {
    get { ZQ(self) } //获取ZQ<Base>实例属性
    set {}
    }
}
extension String: ZQCompatible {}
extension NSString: ZQCompatible {}
extension ZQ where Base: ExpressibleByStringLiteral { //遵守这个协议的不是String就是NSString
    func numberCount() -> Int {
        let string = base as! String
        var count = 0
        for c in string where ("0"..."9").contains(c) {
            count += 1
        }
        return count
    }
    static func read(){
        print("read")
    }
}

//调用
var s1: String = "34DGF443454"
var s2: NSString = "34DGF443454"
var s3: NSMutableString = "34DGF443454"

//类型方法
String.ll.read()//输出:read

print(s1.ll.numberCount())//输出:8
print(s2.ll.numberCount())//输出:8
print(s3.ll.numberCount())//输出:8
  • 引用类型
class Person {}
class Student: Person {}

//让Person遵守这个协议,并且给ZQ前缀扩充方法
extension Person: ZQCompatible {}
extension ZQ where Base: Person {
    func run() {}
    static func test() {}
}

Person.ll.test()
Student.ll.test()

let p = Person()
p.ll.run()

let s = Student()
s.ll.run()
利用协议实现类型判断
  • 传入的是数组实例:
func isArray(_ value: Any) -> Bool { value is [Any] }
isArray( [1, 2] ) //true
isArray( ["1", 2] ) //true
isArray( NSArray() ) //true
isArray( NSMutableArray() ) //true
  • 传入的是数组类型
protocol ArrayType {}
extension Array: ArrayType {}
extension NSArray: ArrayType {}

func isArrayType(_ type: Any.Type) -> Bool { type is ArrayType.Type }
isArrayType([Int].self) //true
isArrayType([Any].self) //true
isArrayType(NSArray.self) //true
isArrayType(NSMutableArray.self) //true

判断某个类型是否遵守某个协议:
类型.self 即为 ArryType.Type
[Int].self 即为 ArryType.Type

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容