函数式编程(Functional Programming)是相对于我们常用的面向对象和面向过程编程的另外一种开发思维方式,它更加强调以函数为中心。善用函数式编程思路,可以对我们的开发工作有很大的帮助和启发,今天我们就来讨论一下吧。

什么是函数式编程

我们用一个简单的例子为大家说明什么是函数式编程。 比如我们有这样一个结构:

struct Staff {

    var firstname: String
    var lastname: String
    var age: Int
    var salary: Float

}

Staff结构定义了员工的基本信息,比如姓名,年龄,薪水等等。 我们再声明一个数组,里面存放我们的员工信息:

let staffs = [
    Staff(firstname:"Ming",lastname:"Zhang",age: 24,salary: 12000.0),Staff(firstname:"Yong",age: 29,salary: 17000.0),Staff(firstname:"TianCi",lastname:"Wang",age: 44,salary: 30000.0),Staff(firstname:"Mingyu",lastname:"Hu",age: 30,salary: 15000.0),Staff(firstname:"TianYun",age: 25,Staff(firstname:"Wang",lastname:"Meng",salary: 14000.0)
]

行为式思维

现在,我们需要找到所有姓张的员工信息(lastname 等于 Zhang),如果用我们惯用的开发思路,可以这样解决:

var staffOfZhang = [Staff]()
for staff in staffs {

    if staff.lastname == "Zhang" {

        staffOfZhang.append(staff)

    }

}

print(staffOfZhang)

这段代码首先声明了一个数组staffOfZhang用于存放符合条件的 Staff 实例,然后我们开始遍历我们的 staffs 数组,对于其中lastname等于 “Zhang” 的实例,将他们添加到staffOfZhang这个数组中。 这就完成了我们这个查找需求。

这种开发思路我们可以称作行为式思路。它侧重于告诉程序如何解决问题。比如我们定义了查询结果存放在哪里,以及如何遍历每一个实例,然后将符合条件的实例读取出来。

声明式思维

解决问题肯定不止一种方法,我们还可以换一种思维方式来解决这个问题,这种思维也可以称为声明式思维。我们可以使用Arrayfilter方法:

let staffOfZhang = staffs.filter { staff in
    return staff.lastname == "Zhang"
}

这就是声明式思维的一个例子,这里staffs.filter函数接受一个闭包类型的参数,filter 方法会对 staffs 中的每一个元素都用传入 filter 的闭包调用一遍,根据这个闭包的返回值决定是否将这个元素作为符合条件(闭包的返回值如果为true则表示符合条件)的元素加入我们的查找结果中。

简单来说我们只需要在传入的闭包中声明好查找的规则,也就是return staff.lastname == "Zhang"这个表达式。这样我们就完成整个查找操作的处理了。

我们这里并没有告诉程序应该怎么去查找满足条件的元素的方法,而只是声明了一个规则。 这样做最大的好处就是能够减少我们的代码量,让我们的代码看起来非常的简洁,而且易理解。 这种方式就是函数式编程的一个例子。

First-Class function

谈到函数式编程就会提到First Class Function。 这是个什么鬼呢~

简单来说 First Class Function 这样的函数不但可以进行简单的调用,还可以赋值给变量,也可以作为参数传递给另外一个函数,还可以作为函数的返回值。关于更详细的描述,可以参看 Wikipedia 上面的这篇文章:https://en.wikipedia.org/wiki/First-class_function

Swift 中的闭包就属于 First Class, 所以我们可以将闭包赋值给变量,传递给函数,作为返回值等等。对于我们上面的那个例子来说,我们就可以这样改写:

func filterZhang(staff:Staff) -> Bool {
    return staff.lastname == "Zhang"
}
staffOfZhang = staffs.filter(filterZhang)

这正好和咱们刚才说的 First Class 对应上了。首先定义了一个 filterZhang 函数,接着将这个函数作为参数传递给 filter。

就这么简单,这也是 Swift 函数式编程的一个体现。

curry

那么,聪明的各位可能又想了,虽然我们可以定义这样一个函数,然后作为参数传递给 filter,但又有什么好处呢? 代码量并不比之前的少。

没错,这个问题问到关键之处了。且听继续分解。函数式编程的另一大特性就是 curry。这也是函数式编程的一个核心概念。

说白了 curry 就是用函数生成另一个函数。关于 curry 的详细探讨,还可以参考我之前的几篇文章:

神奇的 Currying
Swift 中 curry 特性的高级应用

下面咱们就用最简单的例子说明 curry 特性。我们可以将刚才定义的filterZhang方法改写一下:

func filterGenerator(lastnameCondition: String) -> (Staff) -> (Bool) {

    return {staff in
        return staff.lastname == lastnameCondition
    }

}

那么我们来看一下 filterGenerator 的声明,首先它接收一个 String 类型的参数,然后它会返回另一个函数,这个函数接受一个 Staff 类型的参数,并且返回一个布尔值。

详细各位也都看出来了,我们定义的这个新函数 filterGenerator,其实就是对我们之前定义的 filterZhang 函数做了一个更高层的抽象。filterZhang 会过滤处所有 lastname 等于 Zhang 的员工实例。而使用 filterGenerator 可以生成任意条件的过滤函数:

let filterWang = filterGenerator("Wang")
let filterHu = filterGenerator("Hu")

大家看到了吧,我们对 filterGenerator 传入不同的参数,它就会根据这个参数生成一个新的过滤函数。然后我们就可以直接使用:

staffs.filter(filterHu)

这个调用会查询出所有 lastname 等于 Hu 的员工。这就是 curry 特性的精髓。比如,你在开发一个照片处理 APP,会对照片应用很多滤镜,并且这些滤镜还可以叠加,那么使用 curry 方式就可以让你的开发的效率和代码的健壮性提高很多。

更多应用

我们再来多看一些例子。假如我们现在想把所有员工的名字保存到另外一个数组中:

var names = [String]();

for staff in staffs {

    names.append("\(staff.lastname) \(staff.firstname)")

}

print(names)

再来看看我们使用函数式方式如何完成:

names = staffs.map{ staff in
    return "\(staff.lastname) \(staff.firstname)"
}
print(names)

这次我们使用的是 map 方法,它会对数组中所有的元素依照我们指定的规则进行变换,然后生成一个新的数组。这样我们只需要在闭包中声明变换规则就完成了。又是声明式思维的一种体现。

再比如,我们想计算出所有员工的平均工资:

var totalSalary = Float(0.0)

for staff in staffs {

    totalSalary += staff.salary

}
print(totalSalary / Float(staffs.count))

我们再看看如何用函数式的思路来完成:

let averageSalary = staffs.reduce(0) { total,staff in
    return total + staff.salary / Float(staffs.count)
}
print(averageSalary)

这次我们使用 Array 的 reduce 函数,这个函数接受两个参数,第一个参数是初始值,然后 reduce 会依次让每一个元素和这个值进行操作,然后将计算结果传递给下一个元素的调用。

对于我们这里,就是依次计算每个员工对于平均工资的基数,然后将他们相加到一起就是整体的平均工资了。我们这里依然只声明了一个规则return total + staff.salary / Float(staffs.count)。这样代码读起来非常的清晰,很明确的说明了我们要干什么。

结语

到这里,函数式编程的基本思路就都给大家介绍完了。总之呢函数式编程的主要特性就是声明式思维以及 curry 传递思维。它能够让我们用很优雅的语法实现在以前看来比较繁杂的逻辑,这也是它的最大优势。并且它还衍生除了一些分支,比如响应式编程,非常流行的 ReactiveCocoa 库正式 Cocoa 平台对应响应式编程的实现。这些新的编程方式希望用一种更加优雅简洁的方式来解决我们开发中的问题。

最通俗易懂的 Swift 函数式编程的更多相关文章

  1. ios – 如何从Swift中的Closure返回值?

    ).更改实现以使用存储调用值的成员变量.

  2. ios – 如何正确地将选择器作为参数传递给swift

    最后说我有一个包含B的实例的A类.在A类中,我将一个A作为选择器的函数传递给B的一个方法.B使用这个选择器注册通知.但是,当通知进入时,它无法运行选择器并显示“无法识别的选择器发送到实例”.如果我把所有我想在B班做的事情都移到A班,那就行了.不过,我希望他们分开,好像更有条理.我对Objective-C和Swift来说相当新鲜,所以在这种情况下,我不知道如何传递选择器作为参数.斯威夫特答案会很棒.

  3. xcode – Swift闭包:不能将类型'(_) – > Bool’的值转换为预期的参数类型

    注意:使用Xcode7.3.1解决方法一些闪闪发光的东西正在用Swift的类型推断.给卡一个明确的类型,它将工作.您不需要返回类型或返回:注意:这也是:完全指定.Red枚举值也解决了这个问题:在评论中提到,如果您将卡的定义移动到与过滤器相同的文件中,那么它的工作原理.实际上,如果您将CardType枚举从Card中分离出来,只需将CardType移动到带有过滤器的文件中即可.

  4. ios – Swift:如何从不同的swift文件中调用函数

    我的Xcode6beta-2项目中有多个类型为UIViewController的swift文件.我基本上想知道文件A中的一些数据在文件B中使用.我的文件都是UIViewControllers,我创建了一个没有参数的函数,它返回UIViewController_A中的字符串.当我尝试在UIViewController_B中调用所述函数时,intellisense为我填写,但是我必须有一个自动填充为U

  5. ios – Swift – 如何在UITableview中获取所选行的数值

    我在Swift中创建的UITableview上有多个选择,我声明了一个包含所选UITableViewCells的NSIndexPaths的Array.如何将此数组转换为可读术语.例如self.selectedRows是NSloglike:选定项目字符串[{length=2,path=0–1},{length=2,path=0–0},path=0–3}]我希望能够将其转换为:1,2,3.在ObjectiveC中,我通过数组枚举了ObjectsWithOptions,并将数组的id添加到一个可变数组中以获得我

  6. ios – 如何在Swift中使用indicesOfObjectsPassingTest:

    我发现它很混乱,当我第一次在objective-c中学习块时,你如何从声明转换为实际使用方法,而Swift的语法至少是令人困惑的.解决方法这段代码在我的游乐场工作;)希望它有点帮助额外功能定义直列关闭我将上面的示例转换为内联闭包.参数列表和返回类型由术语in分隔.

  7. Swift基础语法汇总

  8. Swift 学习一

    国外开发者最近发现,WWDC2014上苹果发布的新语言Swift,和古老的Scala语言在语法上存在众多的相似之处。Swift语言从语法上来看,几乎是Scala的一个分支,在以下功能上几乎是等同的:类型继承、闭包、元组、协议、扩展、泛型等。不过Swift的运行环境和Scala的区别还是很大,这个概念才是Swift最重要的。但Swift最终编译到机器代码,使用引用计数机制,与Objective-C无缝整合。所以Swift和Scala在代码表象上的相似,应该并不太影响两种语言本质机理上的重大不一致。

  9. Swift基础语法: 24 - Swift的枚举语法, 匹配枚举值和 Switch 语句, 关联值, 原始值

    在前面,我们把函数和闭包解决完毕了,现在让我们来看看Swift中的枚举和Switch:1.枚举语法所谓的枚举其实就是定义一个通用类型的一组相关的值,使你可以在你的代码中以一个安全的方式来使用这些值。

  10. Swift编程高级教程 变量与常量

    提示实际应用中很少需要指定变量数据类型,Swift会根据所设置的值的类型进行推导。Swift使用字符串插值来输出变量和常量。Swift中并没有所谓的实例变量,而是将它们统一为属性了,这样使得属性的声明更加简化。Size封装了宽度和高度。而let只能用于常量的声明,表示它们的值不能发生改变。但是在Swift中可以直接将它们定义为类型的一部分。

随机推荐

  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,所以编译器会报错,现在来一一解决。

返回
顶部