swift 8. 函数
swift 8. 函数标签(空格分隔): swift 今天这一篇学习swift中的函数。函数在任何一门语言中都占据着举足轻重的地位,半边江山都是它的。所以,这一节我会很仔细的学的。 go! 函数的申明和调用函数的申明什么是函数,这个就不用说了吧。有过其他任何一门语言基础的人都应该了解,就不过多解释啥是函数了,函数是一块封闭打包好的代码块,以供其他地方调用。那么swift中function 是如何申明的呢? swift中申明 先举个例子来看如何申明一个函数的: func sayHello(personName: String) -> String {
let greeting = "Hello," + personName + "!"
return greeting
}
看,是不是觉得特别新奇,第一次见到这样子来写一个函数的。 用 看明白了如何申明一个函数了吧,还是蛮简单的,相比其他语言中申明函数的方式,看完上面的之后,反而觉得这种方式还蛮清晰的。它很清晰定义描述了函数做什么,它期望接收什么和执行结束时它返回的结果是什么。 函数的调用上面我们申明了2个函数 我们像这样子调用这2个函数 var name = sayHello("yangyi")
print(name)
那么我们也可以一次性的调用并将结果打印出来: print(sayHello("yangyi"))
值得注意的是:在你申明一个函数的参数是啥类型的时候,已经做了强制申明,所以在你调用这个函数传入的参数就一定是这个类型,不然会报错。 函数的参数与返回值多个参数一般函数,可以有一个参数,也可以有多个参数。和其他的语言一样,多个参数用 func funcName (参数名:参数的类型,参数名:参数的类型...){ } 下面定义了一个三个参数的函数,用于获取个人的信息。 func getInfo(name: String,country: String,age: Int) -> String {
//有3个参数
let info = "Hello,My name is " + name + "!" + " I'm from " + country + " I`m " + String(age) + " old now !"
return info
}
//调用并打印
print(getInfo("yangyi",country:"china",age:18))
//输出:Hello,My name is yangyi! I'm from china I`m 18 old now !
上面这个例子我们用了3个参数,2个 你可能注意到了这行: getInfo("yangyi",country:"china",age:18) 第一个参数直接传入 无参数作为一个函数,肯定是有无参的情况的,即没有参数输入,swift中的函数无参数输入和其他语言一样, func sayHello() -> String {
return "hello"
}
//调用并打印函数
print(sayHello())
//输出:hello
无返回值swift中参数的返回值的类型是通过 func sayHello() {
print("hello")
}
//调用无参数也无返回值的函数
sayHello()
多个返回值一般现代的语言,返回值只有一个,要想返回多个值,要么是封装到一个数组当中返回,要么是对参数取值(&),再返回。所以基本就一个返回值。swift可以允许有多个返回值,借助于 元组在之前就讲过了,是 func funcName (string:String) -> (a:Int,b:Int) { } 我们举例看下如何使用,下面的一个简单的例子,用来查找一个数组中的最小值和最大值。 //多个返回值
func findMaxAndMin(numbers:[Int]) -> (min:Int,max:Int){
var minValue = numbers[0]
var maxValue = numbers[0]
for number in numbers{
minValue = min(minValue,number)
maxValue = max(maxValue,number)
}
return (minValue,maxValue)
}
var scores:[Int] = [1,2,3,4,5,6]
var result = (findMaxAndMin(scores))
print("the min is (result.min)") // 1
print("the max is (result.max)") // 6
上面的例子可能太过简陋,对于传入参数值没做判断,对返回值也没做处理,可能会引起一些错误,我们结合之前学到的一些知识点,对这个函数进行加强,会用到 //多个返回值
func findMaxAndMin(numbers:[Int]) -> (min:Int,max:Int)?{
//判断是否为空,为空则返回nil
if numbers.isEmpty{
return nil
}
//保卫 不为空,否则返回nil
guard !numbers.isEmpty else{
return nil
}
//nil是可选型的标配,返回了nil,所以返回值也得加个?(min:Int,max:Int)?
// 表示返回值是可选型的,可能返回nil。
var minValue = numbers[0]
var maxValue = numbers[0]
for number in numbers{
minValue = min(minValue,maxValue)
}
var scores:[Int]? = [1,6]
scores = scores ?? []; // 用 ?? 确保scores 一定不为nil
if let result = findMaxAndMin(scores!){ // 强制解包,我们已经确定了scores不为nil
print("the min is (result.min)") // 1
print("the max is (result.max)") // 6
}
函数参数名称外部函数参数名我们在写一个函数的时候,使用的参数叫 //定义了2个局部参数a和b
func sayHelloTo(a: String,b: String) -> String {
return "(b),(a)"
}
//调用时,只需要传入对应参数的值就可以了
sayHelloTo("yangyi",b:"hello"); // hello,yangyi
很显然,我并不是很清楚的知道参数a 和 b 是干嘛的。比较难以理解。 所以,现在出现了一种场景,就是我希望强制约束或者告诉调用者,我这几个参数是干嘛的?我想用用文字说明下,那么调用者就能一目了然的知道这几个参数是干嘛的。调用者调用时也必须带上这些参数名。 这就是 func sayHelloTo(name a: String,withWord b: String) -> String {
return "(b),(a)"
}
//调用就加上外部参数名
sayHelloTo(name:"yangyi",withWord:"hello")
这样的好处就是,我们很清晰的告诉外部调用者,我每个参数是干嘛的,而且在我们的函数内部,我们仍然可以使用我们自己定义的比较简单的名字。 由于我们的函数名字 func sayHelloTo(a: String,(a)"
}
sayHelloTo("yangyi",withWord:"hello")
默认条件下,如果我们不去声明第二个,第三个….他们的外部参数,那么内部参数名就是外部参数名。 简写外部参数名上面的外部参数名的书写有点蛋疼,太长了。而且有时候不但没有起到规范和提示的作用,反而还有点碍事。 比如这个例子: func mutiply(num1: Int,num2: Int) -> Int {
return num1*num2
}
上面定义了一个函数 由于,第二个参数,没有加外部参数名,所以,它的内部参数就是外部参数,那么调用就得这样: mutiply(2,num2: 8) // 16
但是,这个函数其实足够简单,就是需要计算2个整型的乘积,这样一弄,调用者蛋疼的不行,估计会骂声一片,能不能省略掉第二个参数的外部名呢?答案是可以的。用 func mutiply(num1: Int,_ num2: Int) -> Int {
return num1*num2
}
mutiply(2,8) // 16
我们由此想到,系统提供给我们的函数,很多其实都是省略了外部参数名的,比如: 参数默认值和其他语言中一样,一个参数是可以有默认值的,当这个参数不传,就会使用默认值。 像这样使用: func sayHiTo(name:String,withWord word:String = "Hi") -> String{
return "(word),(name)"
}
sayHiTo("yangyi") // Hi,yangyi
sayHiTo("yangyi",withWord: "hello") // hello,yangyi
我们用的最多的函数 print(items: Any...,separator: String = " ",terminator: String = "rn") //separator 表示分隔符,就是同时输出几个字符,用什么隔开,默认是空格 //terminator 表示结束符,同时几个print输出时,用什么结束符,默认是rn换行 print(00,66)
print(77,88)
print(12,34,separator: "---",terminator: "!!!")
print(56,78,terminator: "....")
输出就是: 0 66
77 88
12---34!!!56---78....
可变参数可变参数是一个好东西,在写一个函数时,可能有1个参数,可能有2个参数,可能有N个,也可能一个参数也没有。所以这样一来,内参就无法定义了。这个时候,可变参数就起到了很好的作用,很多语言中都有,swift中如何搞呢? 在变量类型名后面加入(…)来定义可变参数。 看这个例子 func addMutil(numbers: Int...) -> Int {
var total: Int = 0
for number in numbers {
total += number
}
return total
}
addMutil(1,4) // 10
addMutil(1,2) // 3
addMutil() // 0
常量参数和变量参数其实,应该叫形参和实参才更靠谱,形参就是在定义函数时,用得参数名字,实参就是调用这个函数传递的参数。在swift中,形参其实是一个 func sayWord(name: String) -> String {
name = name + "hello"
return name
}
你这样子是会报错的: 那有什么办法解决这个问题呢?我就是希望复用这个值,或者需求就是希望这这个参数值上做改动呢? 当有办法:在参数名前面空格加上 func sayWord(var name: String) -> String {
name = name + "hello"
return name
}
这样就可以了。这样name 参数就变成一个变量参数了。可以改变自己的值了。 但是这样子其实只是改变了函数的返回值,并没有改变 name 这个字符串的值,只是我们偷懒在函数里复用了这个name值。 func sayWord(var name: String) -> String {
name = name + "hello"
return name
}
var name = "yangyi"
sayWord("yangyi") //返回yangyihello
name //任然是 yangyi
但是,swift3中会取消 var 这种申明的方式。 输入输出参数用var 什么一个变量参数,仅仅只能在函数体内被更改,而且在swift3中被改变了,那如果你想要一个函数可以修改参数的值,并且想要在这些修改在函数调用结束后仍然存在,那么就应该把这个参数定义为输入输出参数,用关键字 func sayWord(inout name: String) -> String {
name = "hello" + name
return name
}
var myname = "yangyi";
satHello(&myname);
print(myName); //输出:helloyangyi
这样就改变了变量的值。 再看一个例子,用元祖的交换2个变量的值。 func swapTwoNum(inout num1: Int,inout _ num2: Int){
(num1,num2) = (num2,num1)
}
var num1 = 1
var num2 = 2
swapTwoNum(&num1,&num2)
print(num1) // 变成 2
print(num2) // 变成 1
函数类型每个函数都有特定的类型,由参数的类型和返回值的类型决定的。例如,下面的这个个函数: func sayHello(name: String,country: String) -> String {
let info = "Hello," + name + "!" + " I'm from " + country
return info
}
它的函数类似就是 func add (a:Int,_ b:Int) -> Int{
return a + b
}
add 函数的函数类型就是 下面是另一个例子,一个没有参数,也没有返回值的函数: func printHelloWorld() { println("hello,world") } 它是没有参数和返回的,那么它的函数类型就是 使用函数类型知道申明是函数类型之后,我们就可以使用它了,在 Swift 中,使用函数类型就像使用其他类型一样。例如,你可以定义一个类型为函数的常量或变量,并将函数赋值给它: 先定义函数addTwoInts func add (a: Int,_ b: Int) -> Int { return a + b } 再赋值: var mathFunction: (Int,Int) -> Int = add (Int,Int) -> Int 作为一个类型合在一起,表示一个声明的是一个函数变量,这个函数变量有2个整型参数,返回值也是一个整型。 就像前面说过的定义原始的变量字符串String一样: var myapp: String = "iphone" 这个可以读作: “定义一个叫做 mathFunction 的变量,类型是‘一个有两个 Int 型的参数并返回一个 Int 型的值的函数’,并让这个新变量指向 add 函数”。 add 和 mathFunction 有同样的类型,所以这个赋值过程在 Swift 类型检查中是允许的。 所以现在可以这样调用 mathFuntion函数: print("Result: (mathFunction(2,3))") // print "Result: 5" 那么我们如何顶一个一个没有参数,也没有返回值的函数类型呢? func helloWord(){
print( "hello word" )
}
//下面4种方式都是OK的。
var anthorHelloWord1:()->() = helloWord
var anthorHelloWord2:()->Void = helloWord
var anthorHelloWord3:Void->() = helloWord
var anthorHelloWord4:Void->Void = helloWord
函数类型作为参数类型那么,你可能会疑惑了,这样蛋疼的申明一个函数类型,再赋给另一个变量,这样有啥意义呢。其实这样做,就是为了把函数当做一个参数传给给另一个函数。 看下面这个例子: 定义了一个函数:printMathResult ,第一个参数是个函数类型 看这个例子: func printMathResult(mathFunction: (Int,Int) -> Int,_ a: Int,_ b: Int){ print("Result: (mathFunction(a,b))") } printMathResult(mathFunction,5) // 8
这个函数第一个参数mathFunction 是一个(Int,Int) -> Int 类型的函数。在函数里直接打印这个函数调用。 这样,这个函数的实现就由这个函数参数来决定了。 再看另一个例子: var a1:[Int] = [1,7,6]
a1.sort() //排序好了。[1,2,3,4,5,6,7]
a1.sort(isOrderedBefore: (Int,Int) -> Bool) 这个函数类型就是我们自定义排序规则,比如,我们想让大的数排在前面: func sortDesc(a:Int,b:Int)-> Bool{
return a > b
}
a1.sort(sortDesc) // [7,1]
函数类型高阶函数swift中的高阶函数,主要有这几个:
比如,我们希望改变数组中的每一个数组的值,都加2 var array2 = [1,5]
func plus2(num1:Int) -> Int{
return num1 + 2
}
array2.map(plus2) //[3,6,7]
func eg4(num1:Int) -> Bool{
return num1 > 4
}
array2.filter(eg4) // [5]
var array2 = [1,5]
func add (a:Int,_ b:Int) -> Int{
return a + b
}
array2.reduce(0,combine: add) // 8
array2.reduce(0,combine: +) // 8直接用+ 也是表示相加
array2.reduce(0,combine: -) // -15 直接用+ 也是表示相加
array2.reduce(0,combine: *) // 0 因为有元素是0 ,所以相乘永远是0
函数类型作为返回类型函数类型同样可以作为返回值,因为它就被当做一种类型,和普通的类型一样,所以也是可以返回的。你需要做的是在返回箭头 func stepForward(input: Int) -> Int { return input + 1 } func stepBackward(input: Int) -> Int { return input - 1 } 上面分别定义了2个函数,函数类型都是 (Int) -> Int 类型的。一个是加1,一个是减1。 func chooseStepFunction(info: Bool) -> (Int) -> Int { return info ? stepForward : stepForward } 再定义一个函数 所以,我们这样调用: let moveNearerToZero = chooseStepFunction(3 > 0) // moveNearerToZero ==函数 stepForward moveNearerToZero(1) //输出结果是2 let moveNearerToZero = chooseStepFunction(-3 > 0) // moveNearerToZero ==函数 stepBackward moveNearerToZero(1) //输出结果是0 嵌套函数这章中你所见到的所有函数都叫全局函数(global functions),它们定义在全局域中。你也可以把函数定义在别的函数体中,称作嵌套函数(nested functions)。 默认情况下,嵌套函数是对外界不可见的,但是可以被他们封闭函数(enclosing function)来调用。一个封闭函数也可以返回它的某一个嵌套函数,使得这个函数可以在其他域中被使用。 你可以用返回嵌套函数的方式重写 chooseStepFunction 函数: func chooseStepFunction(backwards: Bool) -> (Int) -> Int {
func stepForward(input: Int) -> Int { return input + 1 }
func stepBackward(input: Int) -> Int { return input - 1 }
return backwards ? stepBackward : stepForward
}
var currentValue = -4
let moveNearerToZero = chooseStepFunction(currentValue < 0)
moveNearerToZero(1) // 0
var currentValue = 8
let moveNearerToZero = chooseStepFunction(currentValue < 0)
moveNearerToZero(1) // 9
(编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |