iOS学习笔记42-Swift(二)函数和闭包

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

一、Swift函数

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

/**
* 1、函数第一个参数默认没有外部参数名,其他参数默认有
* 2、可变参数只能在最后一个参数,可变参数的类型是数组
* 3、返回类型也可以是元组
* 4、可以在参数前面加inout关键字,表示内部修改会改变外部的变量,调用时要加“&”符号
* 5、Swift中的函数本身也可以看做一种类型,既可以作为参数又可以作为返回值。
     例如 var fun:(Int,Int)->(Double,Int) = fun2
*/
func 函数名(参数1:类型1, inout 参数2:类型2=默认值2, 可变参数3:类型3...) -> 返回值类型 {

    函数体

    return 返回值
}
函数实例:
1. 普通函数
//1. 定义一个函数,注意参数和返回值,如果没有返回值可以不写返回值或者写成Void、空元组()
func mySum(num1:Int, num2:Int) -> Int{
    return num1 + num2
}
//调用函数,num2是外部参数名,函数第一个参数默认没有外部参数名,其他参数默认有
mySum(1, num2: 2)
2. 设置外部参数名
/**
*  2. 函数参数名分为局部参数名和外部参数名
*/
func mySplit(string a:String, seperator b:Character) -> [String]{
    return ["hello", "world", "!"]
}
/* 
由于给mySplit函数设置了外部参数名string和seperator,所以执行的时候必须带上外部参数名,
此处可以看到一个有意义的外部参数名大大节省开发者使用成本
 */
mySplit(string: "hello,world,!", seperator: ",") //结果:["hello", "world", "!"]
3. 设置默认参数值
//3. 设置函数的最后一个参数默认值设置为",",注意如果使用默认参数那么此参数名将默认作为外部参数名
func mySplit3(string:String, seperator:Character=",")->[String]{
    return ["hello", "world", "!"]
}
mySplit3("hello,world,!") //结果:["hello", "world", "!"]
mySplit3("hello world !", seperator: " ") //结果:["hello", "world", "!"]
4. 设置可变参数
/**
 * 4. 可变参数,一个函数最多有一个可变参数并且作为最后一个参数
 * 下面strings参数在内部是一个[String],对于外部是不定个数的String参数
 */
func myJoinStr(seperator 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:" ", strings: "hello","world","!") //结果:"hello world !"
5. 设置输入输出参数
/**
 *  5. 输入输出参数
 *  通过输入输出参数可以在函数内部修改函数外部的变量(注意调用时不能是常量或字面量)
 *  注意:下面的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: &b) //调用时参数加上“&”符号
print("a=\(a),b=\(b)") //结果:"a=2,b=1"
6. 函数类型使用
/**
 * 6. 函数类型
 */
var sum3 = mySum //自动推断sum3的类型:(Int,Int)->Int,注意不同的函数类型之间不能直接赋值
sum3(1,num2: 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,num2: 2,fn: mySum) //结果:3
caculate(1,num2: 2,fn: 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, num2: b, fn: mySum) //结果:3
myCaculate(a, num2: b, fn: myMinus) //结果:-1
2. 使用闭包
//利用闭包表达式替代函数mySum
myCaculate(a, num2: b, fn: {(num1:Int, num2:Int) -> Int in
    return num1 + num2
}) //结果:3
//利用闭包表达式替代函数myMinus
myCaculate(a, num2: b, fn: {(num1:Int, num2:Int) -> Int in
    return num1 - num2
}) //结果:-1
3. 闭包的简化形式
//简化形式,根据上下文推断类型并且对于单表达式闭包(只有一个语句)可以隐藏return关键字
myCaculate(a, num2: b, fn: { num1, num2 in
    num1 + num2
}) //结果:3
myCaculate(a, num2: b, fn: { num1, num2 in
    num1 - num2
}) //结果:-1
4. 闭包继续简化,使用参数缩写
//再次简化,使用参数名缩写,使用$0...$n代表第n个参数,并且此in关键字也省略了
myCaculate(a, num2: b, fn: { $0 + $1 }) //结果:3
myCaculate(a, num2: b, fn: { $0 - $1 }) //结果:-1

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

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

推荐阅读更多精彩内容