本文来自斯坦福大学的iOS公开课-02
首先举个例子,一个计算器,需要最基本的加减乘除运算,代码如下
import Foundation
//取操作数,假设已经取到如下
let op1 = 1.1;
let op2 = 2.2;
func operate(operation:NSString) {
switch operation {
case "+": performOperation(add)
case "-": performOperation(subtract)
case "×": performOperation(multiply)
case "÷": performOperation(devide)
default: break
}
}
//处理运算
func performOperation(operation:(Double,Double) -> Double) { let value = operation(op1,op2); //得到结果后的一些其他操作 NSLog("value = \(value)") } //× func multiply(op1:Double,op2:Double) -> Double {
return op1 * op2
}
//÷
func devide(op1:Double,op2:Double) -> Double {
return op1 / op2
}
//+
func add(op1:Double,op2:Double) -> Double {
return op1 + op2
}
//-
func subtract(op1:Double,op2:Double) -> Double {
return op1 - op2
}
一个operate
方法用来判断运算符,一个performOperation
方法用来处理运算,四个运算方法加减乘除。
接下来我们来简写。
第一步,省去多余函数
将add
方法体直接替换到operate
方法中。
需要注意三点,
1.直接把函数名之后的从括号开始全部移走
2.左花括号提到函数左括号前面
3.在原来左花括号的位置加一个单词in
//Step 1.直接把函数名之后的从括号开始全部移走
func operate(operation:NSString) {
switch operation {
case "+": performOperation((op1:Double,op2:Double) -> Double {
return op1 + op2
})
...
}
}
//Step 2.左花括号提到函数左括号前面
func operate(operation:NSString) {
switch operation {
case "+": performOperation({(op1:Double,op2:Double) -> Double
return op1 + op2
})
...
}
}
//Step 3.在原来左花括号的位置加一个单词`in`
func operate(operation:NSString) {
switch operation {
case "+": performOperation({(op1:Double,op2:Double) -> Double in
return op1 + op2
})
...
}
}
现在的状态是没有红色报错的,并且改动处仍属于一条语句。
第二步,省去类型
因为Swift最厉害的地方就是类型推断,它可以通过上下文来推断一个变量的类型,所以根据上下文我们知道performOperation
方法的参数为operation:(Double,Double) -> Double
,即一个有两个Double
类型作参数,返回一个Double
类型的函数。
所以,我们可以把类型略掉,把返回类型去掉,箭头也当然要去掉。(这个和Lambda表达式有点类似)
func operate(operation:NSString) {
switch operation {
case "+": performOperation({(op1,op2) in return op1 + op2})
...
}
}
又因为Swift知道里面的表达式一定是返回一个值,所以return
也可以不需要
func operate(operation:NSString) {
switch operation {
case "+": performOperation({(op1,op2) in op1 + op2})
...
}
}
第三步,最简化
这时有人突然跑来看你的代码,觉得很奇怪,为什么这里好像是调用op1
和op2
在做某事,但是你都没有给他们定义呢(其实定义类型已经被我们略去了),那我们就干脆参数名也略掉吧。
在Swift中,会自动给参数顺序命名$0
,$1
,$2
…所以这里也可以把op1
,op2
替换成$0
,$1
,并且函数名也可以完全去掉了。
func operate(operation:NSString) {
switch operation {
case "+": performOperation({$0 + $1})
...
}
}
和原来比简直简化到不能忍!!!
可以回到最前面看下原来的代码,然后再看下下面简化后的:
import Foundation
//取操作数,假设已经取到如下
let op1 = 1.1;
let op2 = 2.2;
func operate(operation:NSString) {
switch operation {
case "+": performOperation({ $0 + $1 })
case "-": performOperation({ $0 - $1 })
case "×": performOperation(){ $0 * $1 } //当参数为最后一个参数时,可以放在外面
case "÷": performOperation{ $0 / $1 } //如果只有它一个参数,甚至可以不需要小括号
default: break
}
}
//处理运算
func performOperation(operation:(Double,Double) -> Double) {
let value = operation(op1,op2);
//得到结果后的一些其他操作
NSLog("value = \(value)")
}
细心的人注意到有个小小的变动,就是花括号刚好为performOperation
最后一个参数,所以花括号可以放在performOperation
的右括号外面;而如果只有一个参数的情况下,小括号可以去掉也没关系。