引子:在上一篇文章中,我们使用到了"Curry"。如果你看了这个框架的源代码的话,可能有点犯晕(有可能只有我一个人这样,大家都是大神)。这篇文章就是关于这个“柯里化”的内容,参考了库作者的博客以及喵神的 tips 和Ole Begemann的文章。顺便说一句喵神出书,可以去支持一下。

什么是柯里化(Currying)

首先,我们来看个简单的例子:

func add(a: Int,b: Int) -> Int{
    return a + b
}

这个函数很容易理解,就是一个整型求和的函数,函数接收两个整型作为参数,并返回两个参数相加的结果。我们可以直接使用该函数:

let sum: Int = add(2,b: 3)  //sum = 5

当我们希望将一个整数数组里面的所有数据都增加一个基数的时候,我们可以进行如下的操作:

let xs = 1...10  
let x = xs.map {
    return add($0,b: 2)
}  
// x = [3,4,5,etc]

在代码中,我们先声明并初始化了一个整型数组,然后用map方法是数组里面的元素都增加了2,并将结果保存到新的数组中。其中$0表示当前迭代到的元素。在代码中,我们使用了闭包(closure),并且传递了一个默认参数。但是当在多处使用类似功能的时候,每次都写闭包还是不够精简。下面做第一步改进:

func addTwo(a: Int) -> Int {
    return add(a,2)
}

let xs = 1...10
let x = xs.map(addTwo)  // x = [3,etc]

但是这个改进有明细的不足,就是默认参数是写死的。当我们要将默认参数设为3的时候我们不得重新写一个函数addThree,这显然不符合代码复用的要求。我们进一步做出修改:

func add(a: Int) -> (Int -> Int) {
    return { b in  
        a + b  
    }  
}

这个函数看起来是不是有点晕啊。我们来一步步分析一下这个函数,首先,函数接收一个整型参数a,函数的返回值是一个Int -> Int 的函数类型,在这个返回的函数类型也会传入一个参数,以及返回一个值(参数a是函数add传入,参数b是闭包里的,返回值是a + b),这意味这两件事:

  1. 如果我们直接调用该函数的化,我们需要使用两个括号来区分这两个参数,形如:

    let sum = add(2)(3)  //sum = 5
  2. 我们可以通过传递一个参数个该函数,那么他会返回一个接收一个参数的新函数。如果我们再传入一个参数给这个返回的无名函数,那么就会返回一个我们想要的结果:

    let addTwo = add(2)
       let xs = 1...100
       let x = xs.map(addTwo) // x = [3,6,etc]

上面的柯里化函数的另外一种更容易理解的形式是:

func add(a: Int)(num: Int) -> Int {
    return a + num
}

//直接调用的形式与上面略有不同:

let sum = add(2)(num: 3) // sum = 5

综上,我们可以将柯里化的进行如下描述:

柯里化就是将一个接收多参数的函数,改造为接收第一个参数,然后返回一个接收余下参数的新方法。柯里化最大的一个好处就是代码复用便于维护,以及量产类似的方法。

//批量产生类似方法:    
let addTwo = add(2)
let addThree = add(3)
let addFour = add(4)

进一步理解柯里化(Currying)

我们看看下面的简单实例:

class BankAccount {
    var balance: Double = 0.0

    func deposit(amount: Double) {
        balance += amount
    }
}

我们最常见的用法:

let account = BankAccount()
account.deposit(100)  // balance is Now 100

下面看一个非常规的用法:

let depositor = BankAccount.deposit
depositor(account)(100)  // balance is Now 200

我们看见下面的操作实现的效果于上面的那个是一样的。让我们来分析一下这个非常规的操作。首先,第一步将函数deposit函数赋值给变量depositor,我们可以看见BankAccount.deposit后面并没有常见的括号。此时的depositor其实是一个BankAccount -> (Double) -> ()类型的实例,有点类似于C中的函数指针。可以在playground右侧看见类型信息,见下图。

BankAccount.deposit()传入参数account其实就是实现绑定。

也就是说depositor传入一个BankAccount类实例作为参数会返回一个Double -> ()类型的函数,再传入一个参数就可以实现变量的叠加了。上面的非常规做法可以进一步简化:

BankAccount.deposit(account)(100) //balance is Now 300

到目前为止,我们只是对一些自己的函数或者自定义类的函数上应用了柯里化,那么如何在系统或者第三方库中应用这种有益的特性呢?我们看下面的例子,主要用到了拓展和泛型。

extension NSNumber {

    class func multiple(left: Int,right:Int) -> Int {
        
        return left * right
        
    }

}

func curry(function: (Int,Int) -> Int) -> (Int -> (Int -> Int)){
    
    return { a in
        
        { b in
            return function(a,b)
        }
        
    }
    
}

首先我们拓展了NSNumber,定义了一个整型的乘法运算函数。然后y又定义了一个curry函数,该函数看起来非常复杂,我们来一步步来分析。首先curry函数接受一个函数类型的参数类型为(Int,Int) -> Int ,然后会返回一个函数类型Int -> (Int -> Int),然后再传入一个整型给该返回类型,回继续返回一个Int -> Int函数类型,再传入参数才会得到最会结果。接下来看函数内部,return语句首先返回了一个闭包,该闭包就是Int -> (Int -> Int)类型:

{ a in 
    //....
 }

再看下一层闭包,类型是(Int -> Int),在最内层的闭包中会将上两层闭包传入的参数a、b进行function操作。但是这并不是最通用的做法,还能改进。可以使用泛型的特征使curry函数不受参数类型限制。具体实现:

extension NSNumber {
    
    class func multiple(left: Int,right:Int) -> Int {
        
        return left * right
            
    }
    
    class func add(left:Double,right:Double) -> Double {
            
        return left + right
            
    }
    
}  

func curry<A,B,C>(function: (A,B) -> C) -> (A -> (B -> C)){
    
    return { a in
                
        { b in
            return function(a,b)
        }
                
    }
    
} 

curry(NSNumber.multiple)(2)(3)      // 6
curry(NSNumber.add)(2.0)(3.0)       // 5.0

具体的类型可以在playground右侧查看。

熟悉了解柯里化的特性后,以后就可以自己写个高复用的类库了,当然也可以使用上篇文章中提到的Curry类库

Swift中的柯里化Currying的更多相关文章

  1. ios – Swift中的UIView动画不起作用,错误的参数错误

    我正在尝试制作动画并使用下面的代码.我得到“无法使用类型’的参数列表调用’animateWithDuration'(FloatLiteralConvertible,延迟:FloatLiteralConvertible,选项:UIViewAnimationoptions,动画:()–>()–>$T4,完成:(Bool)–>(Bool)–>$T5)’“错误.这意味着我使用了错误的参数.我错了.请

  2. ios – 类型不符合协议

    我仍然无法理解Swift中泛型的一些微妙之处.我定义了以下类型:viewForValue现在我定义了以下功能.我希望T是一个符合协议SomeProtocol的UIView.但是,当我执行代码时,我收到以下编译错误:似乎在where子句中我不能指定不实现协议的类.这是为什么?

  3. ios – 将Swift项目转换为易于使用的Cocoapods框架

    编辑1:我更新了项目,因此它已经具有框架结构,我只需要弄清楚我将如何向开发人员提供对自定义部件的访问权限.编辑2:我得到了一个答案,但我认为这个问题可能很容易被误解.目标是使其行为和行为像UICollectionView(与委托,数据源,…

  4. ios – 使用捕获列表中的无主内容导致崩溃,即使块本身也不会执行

    欣赏有关如何调试此内容的任何提示或有关导致崩溃的原因的解释……

  5. ios – Swift传递封闭与Params

    目前我传递一个闭包作为一个对象的属性,该对象不接受参数并且没有返回值,如下所示:到目前为止,这工作得很好.我希望能够在设置此闭包时传入一个参数,以便在MyClass的实例中使用.我正在寻找下面的SOMETHING,虽然我确定语法不正确:我如何将参数传递给可以在MyClass中使用的闭包–即可以在属性本身的didSet部分内使用的值,如第二个示例中所示?

  6. ios – 如何在Swift中解包数组元素? (即数组为数组)

    假设我有一个String数组,我想将它映射到一个Int数组我可以使用map功能:Numbers现在是一个Int?数组,但我想要一个Int数组.我知道我可以这样做:但这似乎并不是很迅速.从Int数组转换?对于ArrayofInt,需要使用相当多的样板函数来调用filter和map.有更快捷的方法吗?解决方法更新:Xcode7.2Swift2.1.1

  7. ios – 来自UIAlertController的self.navigationController?.popViewControllerAnimated

    我是新手,但我想我已经掌握了它.这让我的进步很难过.我想要做的是当我们无法找到他的查询的相关数据时向用户抛出错误消息,然后继续将他带回到之前的ViewController.但是,我在这方面遇到了麻烦.在我添加操作的行上,我收到以下错误:’UIViewController?’不是Void的子类型我该怎么做呢?

  8. ios – Swift:方法重载只在返回类型上有所不同

    我一直在看Swift类,其中定义了两种方法,它们的返回类型不同.我不习惯使用允许这种语言的语言,所以我去寻找描述它如何在Swift中工作的文档.我在任何地方都找不到任何东西.我本来期望在Swift书中有关于它的整个部分.这记录在哪里?

  9. ios – Swift中没有输入参数的通用函数?

    我有一个通用的Swift函数,如下所示:编译器没有错误,但我不知道如何调用此函数.我试过了:但它不起作用.如何在没有输入参数的情况下在Swift中调用Generic函数?解决方法你需要通过一些调用上下文告诉Swift返回类型是什么:注意,在后一种情况下,只有当someCall采用类似于Any的模糊类型作为其参数时,才需要这样做.相反,someCall被指定为[Int]作为参数,函数本身提供上下文,你可以只写someCall事实上,有时可以非常推断出背景!

  10. ios – 如何添加@noescape注释到可选闭包

    我的功能有这个签名:而现在我想让自己在给定的关闭中逃脱自我.但是当我尝试这个:编译器抱怨:是否可以在可选参数中使用它?

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部