上一节我们讲了Swift的基础部分,例如数据类型、运算符和控制流等,现在我们来看下Swift的函数和闭包

一、Swift函数

函数是一个完成独立任务的代码块,Swift中的函数不仅可以像C语言中的函数一样作为函数的参数和返回值,而且还支持嵌套,支持函数参数默认值、可变参数等。

/*
1、在局部参数名前加上#来简写外部参数名(此时局部参数名和外部参数名相同)
2、如果使用默认参数,那么此参数名将默认作为外部参数名(此时局部参数名和外部参数名相同)
3、可变参数只能在最后一个参数,可变参数的类型是数组
4、返回类型也可以是元组
5、可以在参数前面加 var、let、inout 关键字,var表示该局部变量可变,let表示不可变(默认),
inout表示内部修改会改变外部的变量,调用时要加“&”符号
6、Swift中的函数本身也可以看做一种类型,既可以作为参数又可以作为返回值。
例如 var fun:(Int,Int)->(Double,Int) = fun2
*/
func 函数名(var #参数1:类型1, inout 参数2:类型2=默认值2, let #可变参数3:类型3...) -> 返回值类型 { 函数体 return 返回值
}
函数实例:
1. 普通函数
//1. 定义一个函数,注意参数和返回值,如果没有返回值可以不写返回值或者写成Void、空元组()
func mySum(num1:Int, num2:Int) -> Int{
return num1 + num2
}
//调用函数
mySum(1, 2)
2. 设置外部参数名
/*
2. 函数参数名分为局部参数名和外部参数名
*/
func mySplit(string a:String, seperator b:Character) -> [String]{
//调用系统自带的字符串分割函数
return split(a, maxSplit: Int.max, allowEmptySlices: false, isSeparator: {$0==b})
}
/*
由于给mySplit函数设置了外部参数名string和seperator,所以执行的时候必须带上外部参数名,
此处可以看到一个有意义的外部参数名大大节省开发者使用成本
*/
mySplit(string: "hello,world,!", seperator: ",") //结果:["hello", "world", "!"]
3. 快捷设置外部参数名
//3. 下面通过在局部参数名前加上#来简写外部参数名(此时局部参数名和外部参数名相同)
func mySplit2(#string:String, #seperator:Character) -> [String]{
//调用系统自带的字符串分割函数
return split(string, maxSplit: Int.max, allowEmptySlices: false, isSeparator: {$0==seperator})
}
mySplit2(string: "hello,world,!", seperator: ",") //结果:["hello", "world", "!"]
4. 设置默认参数值
//4. 设置函数的最后一个参数默认值设置为",",注意如果使用默认参数那么此参数名将默认作为外部参数名
func mySplit3(#string:String, seperator:Character=",")->[String]{
//调用系统自带的字符串分割函数
return split(string, maxSplit: Int.max, allowEmptySlices: false, isSeparator: {$0==seperator})
}
mySplit3(string: "hello,world,!") //结果:["hello", "world", "!"]
mySplit3(string: "hello world !", seperator: " ") //结果:["hello", "world", "!"]
5. 设置可变参数
/*
5. 可变参数,一个函数最多有一个可变参数并且作为最后一个参数
下面strings参数在内部是一个[String],对于外部是不定个数的String参数
*/
func myJoinStr(seperator:Character=",", strings:String...) -> String{
var result:String = ""
for var i = 0;i < strings.count; ++i{
if i != 0{
result.append(seperator)
}
result += strings[i]
}
return result
}
//调用
myJoinStr(seperator:" ", "hello","world","!") //结果:"hello world !"
6. 设置参数为变量
/*
6. 函数参数默认是常量,不能直接修改,通过声明var可以将其转化为变量(但是注意C语言参数默认是变量)
但是注意这个变量对于外部是无效的,函数执行完就消失了
*/
func mySum2(var num1:Int, num2:Int) -> Int{
num1 = num1 + num2
return num1
}
mySum2(1, 2) //结果:3
7. 设置输入输出参数
/*
7. 输入输出参数
通过输入输出参数可以在函数内部修改函数外部的变量(注意调用时不能是常量或字面量)
注意:下面的mySwap仅仅为了演示,实际使用时请用Swift的全局函数swap
*/
func mySwap(inout a:Int ,inout b:Int){
a = a + b
b = a - b
a = a - b
}
var a = 1,b = 2
mySwap(&a, &b) //调用时参数加上“&”符号
println("a=\(a),b=\(b)") //结果:"a=2,b=1"
8. 函数类型使用
/*
* 8. 函数类型
*/
var sum3 = mySum //自动推断sum3的类型:(Int,Int)->Int,注意不同的函数类型之间不能直接赋值
sum3(1,2) //结果:3 //函数作为返回值
func fn() -> (Int,Int)->Int{
//下面的函数是一个嵌套函数,作用于是在fn函数内部
func minus(a:Int, b:Int) -> Int{
return a - b
}
return minus;
}
var minus = fn()
minus(1, 2) //结果:-1 //函数作为参数
func caculate(num1:Int,num2:Int,fn:(Int,Int)->Int) -> Int{
return fn(num1,num2)
}
caculate(1, 2, mySum) //结果:3
caculate(1,2, minus) //结果:-1

二、闭包

Swift中的闭包其实就是一个函数代码块,它和ObjC中的Block及Java中的lambda是类似的。
闭包的特点就是可以捕获和存储上下文中的常量或者变量的引用,即使这些常量或者变量在原作用域已经被销毁了在代码块中仍然可以使用。

在Swift中闭包表达式的定义形式如下:
{ ( parameters ) -> returnType in
statements;
}
闭包使用:
1. 不使用闭包,使用函数
func mySum(num1:Int,num2:Int) -> Int{
return num1 + num2
}
func myMinus(num1:Int,num2:Int) -> Int{
return num1 - num2
}
func myCaculate(num1:Int, num2:Int, fn:(Int,Int)->Int) -> Int{
return fn(num1,num2)
}
var (a, b) = (1, 2)
myCaculate(a, b, mySum) //结果:3
myCaculate(a, b, myMinus) //结果:-1
2. 使用闭包
//利用闭包表达式替代函数mySum
myCaculate(a, b, {(num1:Int, num2:Int) -> Int in
return num1 + num2
}) //结果:3
//利用闭包表达式替代函数myMinus
myCaculate(a, b, {(num1:Int, num2:Int) -> Int in
return num1 - num2
}) //结果:-1
3. 闭包的简化形式
//简化形式,根据上下文推断类型并且对于单表达式闭包(只有一个语句)可以隐藏return关键字
myCaculate(a, b, { num1, num2 in
num1 + num2
}) //结果:3
myCaculate(a, b, { num1, num2 in
num1 - num2
}) //结果:-1
4. 闭包继续简化,使用参数缩写
//再次简化,使用参数名缩写,使用$0...$n代表第n个参数,并且此in关键字也省略了
myCaculate(a, b, { $0 + $1 }) //结果:3
myCaculate(a, b, { $0 - $1 }) //结果:-1

考虑到闭包表达式的可读取性,Swift中如果一个函数的最后一个参数是一个函数类型的参数(或者说是闭包表达式),则可以将此参数写在函数括号之后,这种闭包称之为“尾随闭包”

5. 尾随闭包
//尾随闭包
myCaculate(a, b) {
$0 + $1
} //结果:3
myCaculate(a, b) {
$0 - $1
} //结果:-1
6. 捕获变量

前面说过闭包之所以称之为“闭包”,就是因为其可以捕获一定作用域内的常量或者变量进而闭合并包裹着。

func myAdd() -> ()->Int {
var total = 0
var step = 1
func fn() -> Int{
total += step
return total
}
return fn
}
/*
fn捕获了total和step,尽管下面的myAdd()执行完后total和step被释放,
但是由于fn捕获了二者的副本,所以fn会随着两个变量的副本一起被存储
*/
var a = myAdd()
a() //结果:1
a() //结果:2,说明a中保存了total的副本(否则结果会是1) var b = myAdd()
b() //结果:1,说明a和b单独保存了total的副本(否则结果会是3) var c = b
c() //结果:2,说明闭包是引用类型,换句话说函数是引用类型(否则结果会是1)
  • Swift会自动决定捕获变量或者常量副本的拷贝类型(值拷贝或者引用拷贝)而不需要开发者关心
  • 被捕获的变量或者常量的内存管理同样是由Swift来管理,我们不用关心,例如当上面的函数a不再使用了,那么fn捕获的两个变量也就释放了。

iOS学习笔记42-Swift(二)函数和闭包的更多相关文章

  1. iOS学习笔记13-网络(二)NSURLSession

    在2013年WWDC上苹果揭开了NSURLSession的面纱,将它作为NSURLConnection的继任者.现在使用最广泛的第三方网络框架:AFNetworking.SDWebImage等等都使用 ...

  2. iOS学习笔记20-地图(二)MapKit框架

    一.地图开发介绍 从iOS6.0开始地图数据不再由谷歌驱动,而是改用自家地图,当然在国内它的数据是由高德地图提供的. 在iOS中进行地图开发主要有三种方式: 利用MapKit框架进行地图开发,利用这种 ...

  3. iOS学习笔记(十二)——iOS国际化

    开发的移动应用更希望获取更多用户,走向世界,这就需要应用国际化,国际化其实就是多语言.这篇文章介绍Xcode4.5以后的国际化,包括应用名国际化和应用内容国际化.如果是Xcode4.5之前版本请参考. ...

  4. Python3学习笔记(十二):闭包

    闭包定义: 在一个外函数中定义了一个内函数,内函数里引用了外函数的临时变量,并且外函数的返回值是内函数的引用.这样就构成了一个闭包. 我们先来看一个简单的函数: def outer(a): b = 1 ...

  5. js学习笔记之自调用函数、闭包、原型链

     自调用函数 var name = 'world!'; // console.log(typeof name) (function () { console.log(this.name, name, ...

  6. IOS学习笔记06---C语言函数

    IOS学习笔记06---C语言函数 --------------------------------------------  qq交流群:创梦技术交流群:251572072              ...

  7. IOS学习笔记07---C语言函数-printf函数

    IOS学习笔记07---C语言函数-printf函数 0 7.C语言5-printf函数 ------------------------- ----------------------------- ...

  8. React Native 学习笔记--进阶(二)--动画

    React Native 进阶(二)–动画 动画 流畅.有意义的动画对于移动应用用户体验来说是非常必要的.我们可以联合使用两个互补的系统:用于全局的布局动画LayoutAnimation,和用于创建更 ...

  9. iOS学习笔记-精华整理

    iOS学习笔记总结整理 一.内存管理情况 1- autorelease,当用户的代码在持续运行时,自动释放池是不会被销毁的,这段时间内用户可以安全地使用自动释放的对象.当用户的代码运行告一段 落,开始 ...

随机推荐

  1. fpga Verilog hdl 按键消抖 部分程序讲解

    module debounce(clk_in,rst_in,key_in,key_pulse,key_state); input clk_in;//system clock input rst_in; ...

  2. windows7桌面小工具打不开的解决方案

    将任务管理器中的sidebar.exe结束任务: 将C:\Users\用户名\AppData\Local\Microsoft\Windows Sidebar下的settings.ini的文件名修改为任 ...

  3. [OpenMP] 并行计算入门

    OpenMP并行计算入门 个人理解 OpenMP是一种通过共享内存并行系统的多处理器程序设计的编译处理方案,通过预编译指令告诉编译器哪些代码块需要被并行化,通过拷贝代码块实现并行程序.对于循环的并行化 ...

  4. 解决ubuntu上ifconfig没有eth0/ens33且无法上网的问题

    ifconfig只有一个轮回端口lo,没有我们的网卡eth0,一开始以为是vsphere(新手对于vsphere不是很熟悉)上我的虚拟机配置问题,还查看了相关的网络配置,后来才知道是因为: 问题出在配 ...

  5. 【转载】Alpha、Beta、RC、GA版本的区别

    转自:http://www.blogjava.net/RomulusW/archive/2008/05/04/197985.html Alpha:是内部测试版,一般不向外部发布,会有很多Bug.一般只 ...

  6. JS数据结构及算法(一) 堆栈

    最近在看<学习JavaScript数据结构与算法>这本书,感觉自己又涨知识了 哈哈... 现在将自己看的做个总结,也是巩固理解. 栈:先进后出,新添加和待删除的元素都保存在栈顶.可以用数组 ...

  7. 洛谷 P1147 连续自然数和

    洛谷 P1147 连续自然数和 看到dalao们的各种高深方法,本蒟蒻一个都没看懂... 于是,我来发一篇蒟蒻友好型的简单题解 #include<bits/stdc++.h> using ...

  8. python从列表中删除相邻重复元素

    这里以一个栗子来用三种方法实现,输入a=['1','1','2','2','1','1'],输出b=['1', '2', '1'] 方法一: list1 = ['] def del_adjacent( ...

  9. 01Qt中的隐式共享

    隐式共享 ​ 隐式共享又称为回写复制(copy on write).当两个对象共享同一分数据时(通过浅拷贝实现数据共享),如果数据不改变,则不进行数据的复制.而当某个对象需要需要改变数据时,则进行深拷 ...

  10. Tomcat上传文件报错:returned a response status of 403 Forbidden

    出现这样的错误是没有权限对服务器进行写操作.需要在这个项目所在的tomcat中配置可写操作即可: 在tomcat的web.xml添加下面代码: <init-param><param- ...