Swift # 柯里化函数

前言

  此次文章,讲述的是Swift的一个新特性(柯里化函数),可能很多iOS开发人员是第一次听这个词汇,包括我自己也是,自己也用了几天时间才总结出来,希望能帮助到各位咯,个人感觉偏向有开发经验的码友,如果零基础的看懂,希望能给个赞,??!

如果喜欢我的文章,可以关注我,随着后续不断学习Swift中,陆续还会有更新ing....

什么是柯里化函数?

  柯里化(Currying),又称部分求值(Partial Evaluation),是一种函数式编程思想,就是把接受多个参数的函数转换成接收一个单一参数(最初函数的第一个参数)的函数,并且返回一个接受余下参数的新函数技术。

class Currying

{

    // uncurried:普通函数

    // 接收多个参数的函数(与类相关的函数,统称为方法,但是这里就直接说函数了,方便理解)

    func add(a: Int, b: Int, c: Int) -> Int{

        println("\(a) + \(b) + \(c)")

        return a + b + c

    }

    // curried:柯里化函数

    // 柯里化函数,Swift中已经支持这样的语法了,可以直接写

    func addCur(a: Int)(b: Int)(c: Int) -> Int{

        println("\(a) + \(b) + \(c)")

        return a + b + c

    }

}

如何定义柯里化函数?

如图定义柯里化函数:

柯里化函数实现原理

class Currying

{

    /*** uncurried:普通函数 ***/

    // 接收多个参数的函数

    func add(a: Int, b: Int, c: Int) -> Int{

        println("\(a) + \(b) + \(c)")

        return a + b + c

    }

    /*** 手动实现柯里化函数 ***/

    // 把上面的函数转换为柯里化函数,首先转成接收第一个参数a,并且返回接收余下第一个参数b的新函数(采用闭包)

    // 为了让大家都能看懂,我帮你们拆解来看下

    // (a: Int) : 参数

    // (b:Int) -> (c: Int) -> Int : 函数返回值(一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数)

    // 定义一个接收参数a,并且返回一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数

    func add(a: Int) -> (b:Int) -> (c: Int) -> Int{

        // 一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数

        return { (b:Int) -> (c: Int) -> Int in

         // 返回一个接收余下第一个参数c,并且有返回结果为Int类型的函数

         return { (c: Int) -> Int in

         return a + b + c;

     注解: 这里为什么能使用参数a,b,c?

           利用闭包的值捕获特性,即使这些值作用域不在了,也可以捕获到他们的值。

           闭包会自动判断捕获的值是值拷贝还是值引用,如果修改了,就是值引用,否则值拷贝。

           注意只有在闭包中才可以,a,b,c都在闭包中。

            }
        }
    }

    /*** curried: 系统自带的柯里化函数 ***/

    func addCur(a: Int)(b: Int)(c: Int) -> Int{

        println("\(a) + \(b) + \(c)")

        return a + b + c

    }
}

如何调用柯里化函数

由于定义的是一个实例方法,因此调用需要依赖对象.

如图调用柯里化函数:

// 创建柯里化类的实例
var curryInstance = Currying()

/*** 调用手动实现的柯里化函数 **/

var r: Int = curryInstance.add(10)(b: 20)(c: 30)

// 可能很多人都是第一次看这样的调用,感觉有点不可思议。

// 让我们回顾下OC创建对象 [[Person alloc] init],这种写法应该都见过吧,就是一下发送了两个消息,alloc返回一个实例,再用实例调用init初始化,上面也是一样,一下调用多个函数,每次调用都会返回一个函数,然后再次调用这个返回的函数。

/***** 柯里化函数分解调用 *****/

// 让我来帮你们拆解下,更容易看懂

// curryInstance.add(10): 调用一个接收参数a,并且返回一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数

// functionB: 一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数
let functionB = curryInstance.add(10)

// functionB(b: 20):调用一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数

// functionC: 一个接收参数c,返回值为Int类型的函数
let functionC = functionB(b: 20)

// functionC(c: 30): 调用一个接收参数c,返回值为Int类型的函数

// result: 函数的返回值
var res: Int = functionC(c: 30);

// 这里会有疑问?,为什么不是调用curryInstance.add(a: 10),而是curryInstance.add(10),functionB(b: 20),functionC(c: 30),怎么就有b,c,这是因为func add(a: Int) -> (b:Int) -> (c: Int) -> Int这个方法中a是第一个参数,默认是没有外部参数名,只有余下的参数才有外部参数名,b,c都属于余下的参数。

/***** 系统的柯里化函数调用 *****/
var result: Int = curryInstance.addCur(10)(b: 20)(c: 30)

/***** 系统的柯里化函数拆解调用 *****/

// 注意:Swift是强类型语言,这里没有报错,说明调用系统柯里化函数返回的类型和手动的functionB类型一致

// curryInstance.addCur(10) : 调用一个接收参数a,并且返回一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数

// functionB: 一个接收参数b的函数,并且这个函数又返回一个接收参数c,返回值为Int类型的函数
functionB = curryInstance.addCur(10)

// functionC: 一个接收参数c,返回值为Int类型的函数
functionC = functionB(b: 20)

// result: 函数的返回值
res = functionC(c: 30)

// 打印 60,60,60说明手动实现的柯里化函数,和系统的一样。

println("\(r),\(res),\(result)")

柯里化函数使用注意

1必须按照参数的定义顺序来调用柯里化函数,否则就会报错。

var result: Int = curryInstance.addCur(10)(b: 20)(c: 30)

2、柯里化函数的函数体只会执行一次,只会在调用完最后一个参数的时候执行柯里化函数体。以下调用functionC(c: 30)才会执行函数体。这个可以自己断点调试

// curried:柯里化函数
func addCur(a: Int)(b: Int)(c: Int) -> Int{

    println("\(a) + \(b) + \(c)")

    return a + b + c
}

// 创建柯里化类的实例
var curryInstance = Currying()

// 不会执行柯里化函数体
functionB = curryInstance.addCur(10)

// 不会执行柯里化函数体
functionC = functionB(b: 20)

// 执行柯里化函数体
res = functionC(c: 30)

Swift中实例方法就是一个柯里化函数

如何获取实例方法?可以直接通过类获取实例方法.

注意:方法是什么类型,就返回什么类型的函数,不过需要传入一个参数(类实例)才能获取到,如果方法中有外部参数名,外部参数名也属于类型的一部分

如图使用类获取实例方法:

Swift中实例方法的另一种调用方式(柯里化调用)

// 创建柯里化类的实例

var curryInstance = Currying()

// 调用function方法

Currying.function(curryInstance)()

// 拆解调用function方法

// 1.获取function方法

let function = Currying.function(curryInstance)

// 2.调用function方法

function()

注解: 步骤都是一样,首先获取实例方法,在调用实例方法,实例方法怎么调用,就不需要在教了。

柯里化函数有什么好处?为什么要使用它?

这里就需要了解函数式编程思想了,推荐看这篇文章函数式编程初探

特点:

1.只用“表达式”(表达式:单纯的运算过程,总是有返回值),不用“语句”(语句:执行某种操作,没有返回值)。2.不修改值,只返回新值。

好处:

1.代码简洁

2.提高代码复用性

3.代码管理方便,相互之间不依赖,每个函数都是一个独立的模块,很容易进行单元测试。

4.易于“并发编程”,因为不修改变量的值,都是返回新值。

柯里化函数就是运用了函数式编程思想,因此它也有以上的好处。

在iOS开发中如何运用柯里化函数(实用性)

实用性一:复用性

需求1:地图类产品,很多界面都有相同的功能模块,比如搜索框。

我们可以利用柯里化函数,来组装界面,把界面分成一个个小模块,这样其他界面有相同的模块,直接运用模块代码,去重新组装下就好了。

实用性二:延迟性,柯里化函数代码需要前面的方法调用完成之后,才会来到柯里化函数代码中。

需求2:阅读类产品,一个界面的显示,依赖于数据,需要加载完数据之后,才能判断界面显示。

这时候也可以利用柯里化函数,来组装界面,把各个模块加载数据的方法抽出来,等全部加载完成,在去执行柯里化函数,柯里化函数主要实现界面的组装。

举例说明:

// 组合接口

// 为什么要定义接口,为了程序的扩展性,以后只需要在接口中添加对应的组合方法就好了。

protocol CombineUI

{
    func combine(top: () -> ())(bottom: () -> ())()
}

// 定义一个界面类,遵守组合接口
class UI: CombineUI

{
    func combine(top: () -> ())(bottom: () -> ())() {

        // 搭建顶部
        top()

        // 搭建底部
        bottom()
    }
}

点击下载:源代码

来源:啊崢的简书

微博:吖了个峥,欢迎交流。

|--> Copyright (c) 2015 Bing Ma.

|--> GitHub RUL: https://github.com/SpongeBob-GitHub

时间: 2024-12-26 12:34:17

Swift # 柯里化函数的相关文章

javascript中利用柯里化函数实现bind方法

柯理化函数思想:一个js预先处理的思想:利用函数执行可以形成一个不销毁的作用域的原理,把需要预先处理的内容都储存在这个不销毁的作用域中,并且返回一个小函数,以后我们执行的都是小函数,在小函数中把之前预先存储的值进行相关的操作处理即可: 柯里化函数主要起到预处理的作用: bind方法的作用:把传递进来的callback回调方法中的this预先处理为上下文context; /** * bind方法实现原理1 * @param callback [Function] 回调函数 * @param con

浅谈JavaScript中的柯里化函数

首先,不可避免的要引经据典啦,什么是柯里化函数呢(from baidu): 在计算机科学中,柯里化(Currying)是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数且返回结果的新函数的技术.这个技术由 Christopher Strachey 以逻辑学家 Haskell Curry 命名的,尽管它是 Moses Schnfinkel 和 Gottlob Frege 发明的. 用于创建已经设置好了一个或多个参数的函数 与函数绑定相似,他们之间的区

JavaScript的柯里化函数

柯里化,或者说部分应用,是一种函数式编程的技术,对于熟悉以传统方式编写 JavaScript 代码的人来说可能会很费解.但如果使用得当,它可以使你的 JavaScript 函数更具可读性. 更具可读性和灵活性 函数式 JavaScript 被吹捧的优点之一就是拥有短小紧凑的代码风格,可以用最少行数.更少重复的代码得到正确的结果.有时这会以牺牲可读性为代价:如果你还不熟悉函数式编程的方法,这种方法写的代码会很难阅读和理解. 如果之前你遇到过柯里化这个术语,但是不知道它是什么意思,把它当做奇怪的.难

柯里化函数之Javascript

柯里化函数之Javascript 定义 根据定义来说,柯里化就是将一个接收"多个"参数的函数拆分成一个或者许多个接收"单一"参数的函数.定义看起来是比较抽象的,下面来举个例子: 代码 1 2 3 4 5 function concat(str1,str2){ return str1 + str2; } concat("forever","px") // "foreverpx" 不难理解,上面的代码中定义了一

节流函数&防抖函数 柯里化函数

/* onscroll onresize input ..... 节流函数 让高频率事件进行减少触发变成低频率事件 var bStop = true; window.onscroll = function() { if(!bStop){ return; } bStop = false; setTimeout(()=>{ var t = document.documentElement.scrollTop || document.body.scrollTop; console.log(t); bS

柯里化函数、快速排序、外边距重叠

柯里化函数柯里化函数通常也称为部分求值,其含义是给函数分步传递参数,每次传递参数后部分应用参数,并返回一个更具体的函数接受剩下的参数,这中间可嵌套多层这样的接受部分函数参数,直至返回最后结果. 例子:实现carrying函数,完成以下需求,代码如下:function sum(a,b,c){return a+b+c;}function currying(fn){ `//todo`(自己编写的代码) }var curring = curring(sum);console.log(currying(s

柯里化函数的实现

记录柯里化函数实现的学习过程: 柯里化通常也称部分求值,其含义是给函数分步传递参数,每次传递参数后部分应用参数,并返回一个更具体的函数接受剩下的参数,这中间可嵌套多层这样的接受部分参数函数,直至返回最后结果. 如果要实现下面这个方法: add(2)(1, 3, 4)(2, 3)(3)(4, 6)(7, 98)() // 133 上面这个函数当参数为空的时候执行了内部参数所有值的相加,所以我们应该考虑当参数不为空的时候将缓存起来,在为空的时候再相加,这样的思路会用闭包的方式来实现.下面是实现方法:

关于柯里化函数

柯里化(Currying)是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数而且返回结果的新函数的技术. 特点: 参数复用 – 复用最初函数的第一个参数 提前返回 – 返回接受余下的参数且返回结果的新函数 延迟执行 – 返回新函数,等待执行 在编程开发中,使用柯里化函数封装解决问题的例子主要有: 兼容浏览器事件监听方法 性能优化:防抖和节流 兼容低版本IE的bind方法 例:设计一个柯里化函数(currying ),实现sum(100,200)(3

swift 柯里化

如何将一个接受多参数的函数变换为一系列只接受单个参数的函数,这个过程被称为柯里化 (Currying) 参照:objc.io|objc 中国 /** 1.add0 和 add的例子向我们展示了如何将一个接受多参数的函数变换为一系列只接受单个参数的函数,这个过程被称为柯里化 (Currying),它得名于逻辑学家 Haskell Curry;我们将 add 称为 add0 的柯里化版本 2.首先向函数传递第一个参数 1,然后将返回的函数应用到第二个参数 2 3.我们甚至可以省略 addTheSom