Swfit的學(xué)習(xí)大致有幾個(gè)重點(diǎn):
元組,可選興,函數(shù),閉包
這些是較之OC有很大不同的地方婶芭,學(xué)會(huì)了這些swift也算是入門了
閉包有三種形式:
全局函數(shù):有名字但不能捕獲任何值。
嵌套函數(shù):有名字着饥,也能捕獲封閉函數(shù)內(nèi)的值犀农。
閉包表達(dá)式:無(wú)名閉包,使用輕量級(jí)語(yǔ)法宰掉,可以根據(jù)上下文環(huán)境捕獲值呵哨。
捕獲值
1、閉包可以在其定義的上下文中捕獲常量或變量轨奄。 即使定義這些常量和變量的原域已經(jīng)不存在仇穗,閉包仍然可以在閉包函數(shù)體內(nèi)引用和修改這些值。我認(rèn)為可以理解為捕獲增加了原常量或變量的引用計(jì)數(shù)戚绕。
2纹坐、Swift最簡(jiǎn)單的閉包形式是嵌套函數(shù),也就是定義在其他函數(shù)的函數(shù)體內(nèi)的函數(shù)舞丛。嵌套函數(shù)可以捕獲其外部函數(shù)所有的參數(shù)以及定義的常量和變量耘子。
func makeIncrementer(forIncrement amount: Int) -> () -> Int {
var runningTotal = 0
func incrementer() -> Int {
runningTotal += amount
return runningTotal
}
return incrementer
}
let a = makeIncrementer(forIncrement:10)
print(a())
全局函數(shù)
什么叫全局函數(shù)?全局函數(shù)不能捕獲值球切?啥意思呢谷誓?一陣懵逼。
func testGlobalFunc()
{
//這不是捕獲了外部的value嗎吨凑?
print(value);
}
testGlobalFunc()
我的理解是這樣的:
全局函數(shù)是定義在類之外的不屬于任何類的函數(shù)捍歪。
寫OC的時(shí)候常常會(huì)用到各種宏定義,但是Swift中沒(méi)有宏,通常是通過(guò)全局常量或者全局函數(shù)來(lái)實(shí)現(xiàn)這一效果.我們只需要建一個(gè)文件(假設(shè)叫Global.swift),把想用的定義在里面,無(wú)須導(dǎo)入頭文件什么的,就可以在全局用啦.
在類中實(shí)現(xiàn)的函數(shù)是成員函數(shù)户辱,也就是我們一般說(shuō)的函數(shù)
//
// Global.swift
//
// Created by apple on 2020/3/29.
//
import Foundation
import UIKit
/**
* 替代oc中的#define,列舉一些常用宏
*/
let kScreenWidth = UIScreen.main.bounds.size.width
// 屏幕的物理寬度
func kScreenWidthFun() ->CGFloat
{
return kScreenWidth
}
func kScreenHeighFun() ->CGFloat
{
return UIScreen.main.bounds.size.height
}
一、閉包引入
普通函數(shù)寫法
func square(num:Int)->Int{
return num*num
}
print(square(num:3))
閉包寫法
let fun1 = {
(num:Int)->Int in
return num*num
}
print(type(of:fun1))//fun1類型:(Int) -> Int
print(fun1(4))
閉包表達(dá)式語(yǔ)法
1糙臼、由一對(duì){}開(kāi)始和結(jié)束庐镐;
2、in關(guān)鍵字把閉包分成兩部分:參數(shù)與返回值变逃、閉包體
思考1:demo是不是一個(gè)閉包
let demo = {print("hello")}
print(type(of:demo))// () -> ()
第二部分:閉包縮寫
需求:把我們的java成績(jī)放到了一個(gè)數(shù)組里必逆,定義一個(gè)函數(shù),把大于某個(gè)閾值的成績(jī)返回給我
func getScore(score:[Int],con:(Int)->Bool)->[Int]{
var newScore = [Int]()
for item in score{
if con(item){
newScore.append(item)
}
}
return newScore
}
var score = [66,89,80,33,100]
print(getScore(score:score,con:{(s:Int)->Bool in return s>40}))
省略1:省略->返回類型(自動(dòng)推斷出返回值是一個(gè)Bool)
print(getScore(score:score,con:{(s:Int) in return s>40}))
省略2:省略參數(shù)類型和括號(hào)(自動(dòng)推斷出參數(shù)類型是Int)
print(getScore(score:score,con:{s in return s>40}))
省略3:?jiǎn)涡斜磉_(dá)式閉包可以省略 return 關(guān)鍵字來(lái)
print(getScore(score:score,con:{s in s>40}))
省略4:省略參數(shù)列表定義揽乱,用1等等指代參數(shù)名眉,同時(shí)省略in
print(getScore(score:score,con:{$0>40}))
第三部分:尾隨閉包
使用情景:當(dāng)閉包表達(dá)式作為最后一個(gè)參數(shù)傳遞給函數(shù)時(shí),可以單獨(dú)提出來(lái)
問(wèn)題1:尾隨指的是跟在別人后面 閉包到底跟在誰(shuí)后面凰棉?-函數(shù)后面
問(wèn)題2:用尾隨閉包的好處损拢?提升代碼的可讀性
func printInfo(info:String,printFun:(String)->Void){
printFun(info)
}
//普通調(diào)用方式
printInfo(info:"hello world",printFun:{s in print(s+"~~~")})
//使用尾隨閉包進(jìn)行調(diào)用
printInfo(info:"hello world"){s in print(s+"~~~")}
閉包是引用類型
let b = a
print(b()) //輸出結(jié)果為20
let c = makeIncrementer(forIncrement:10)
print(c())
逃逸閉包
閉包要出逃,出逃到哪里呢撒犀?逃到函數(shù)外使用探橱!牛掰!
需求:閉包作為一個(gè)參數(shù)傳遞一個(gè)函數(shù)绘证,但是這個(gè)閉包我不立馬使用,先把這個(gè)閉包存起來(lái)哗讥,過(guò)會(huì)再用
var recv:()->Void = {print("")}
var x = 10
//方案一:定義一個(gè)函數(shù)嚷那,接受一個(gè)普通閉包為參數(shù)
func test1(closure:()->Void){
recv = closure //此段代碼報(bào)錯(cuò),原因是普通閉包作為參數(shù)杆煞,會(huì)在函數(shù)結(jié)束之后被銷毀魏宽,無(wú)法在函數(shù)外使用。
}
test1{
x=100
}
recv()
// 方案二:逃逸閉包
/*
逃逸閉包特點(diǎn)如下:
1决乎、可以在函數(shù)結(jié)束后使用队询;
2、壽命長(zhǎng)构诚!逃逸閉包聲明周期長(zhǎng)于函數(shù)蚌斩,只要它的引用被其他對(duì)象持有,就不會(huì)隨著函數(shù)結(jié)束而釋放掉
3范嘱、通過(guò)@escaping 指定一個(gè)閉包是逃逸閉包
*/
func test2(closure:@escaping ()->Void){
recv = closure
}
test2{
x = 200
}
recv()
print(x)
逃逸閉包在網(wǎng)絡(luò)訪問(wèn)中會(huì)經(jīng)常用到送膳!