第一部分: 基础部分 ~ 闭包函数

2022-07-02  本文已影响0人  wg689
import UIKit
import Darwin

var greeting = "Hello, playground"

let a = 10

// 10:57
let (c, b) = (10, 11)

//Swift 的赋值操作并不返回任何值。所以下面语句是无效的:

let f = 10 % 9

let three = 3

let minusThree = -three

let alsoThree = +three

var s = 2

s += 2

1 == 1

2 != 1

let name = "word"

if name == "word" {
    print("equal")
} else {
    print("\(name) not equal")
}


(1, "zebe") == (1, "zebe")
(1, "zebe") == (2, "zebe")

(2, "c") < (2, "d")



let contentHeight = 12

let hasHeader = false

var rowHeight = contentHeight

if hasHeader {
    rowHeight = rowHeight + 20
} else {
    rowHeight = rowHeight - 20
}

rowHeight = hasHeader ? rowHeight + 20 : rowHeight - 20


let defaultColorName = "red"

let userDefinedColorName: String? = ""

var colorNameToUse = userDefinedColorName ?? defaultColorName

for index in 1...5 {
    print("\(index) * 5 = \(index * 5) ")
}

// 闭区间
let names = ["11","22","33","44","55"]
let  count = names.count
for i in 0 ..< count {
    print("第\(i)个人的名字叫做 \(names[i])")
}


for name in names[2...] {
    print(name)
}

for name in names[...2] {
    print(name)
}



for name in names[..<2] {
    print(name)
}


let allowedEntry = false
if !allowedEntry {
    print("Aceess")
}

let enteredDoorCode = true
let passedRetinaScan = false

if enteredDoorCode && passedRetinaScan {
    print("welcome")
} else {
    print("ass denit")
}




if enteredDoorCode || passedRetinaScan {
    print("welcome")
} else {
    print("ass denit")
}


if (enteredDoorCode && passedRetinaScan) || hasHeader  {
    print("Welcome!")
} else {
    print("ACCESS DENIED")
}



// 字符串和字符
let someString = "Some string literal value"

var quatation = """
   Some string literal \
     Some string literal
     Some string literal
   Some string literal
"""
print(quatation)

quatation = """
Some string literal 2
"""
print(quatation)


// 字符串字面量的特殊字符

let wiseWords = "\"Imagination is more important than knowledge\" - Einston"

print(wiseWords)

var  dollarSign = "\u{24}"
print(dollarSign)

dollarSign = "\u{2665}"
print(dollarSign)

let threeDoubleQuotes = """
"Escaping the first quote"
"Escaping the first quote"\"\"
"""

print(threeDoubleQuotes)

//扩展字符串分隔符
let threeMoreDoubleQuotationMarks = #"""
Here are three more double quotes: """
"""#
print(threeMoreDoubleQuotationMarks)


var emptyString = ""
var emptyString2 = String()

if emptyString2.isEmpty {
    print("is emptyString")
}


var viriableString = "Horse"

viriableString +=  "and carriage"

for charactor in "Dog!" {
    print(charactor)
}


let string1 = "hello"
let string2 = "hello"
var welcome = string1 + string2

let exclamationMark: Character = "!"
welcome.append(exclamationMark)


let badStart = """
one
two

"""

let end = """
three
"""

print(badStart + end)


/// 字符串差值
let multiplier = 3
//// message 是 "3 times 2.5 is 7.5"

let message = "\(multiplier) times 2.5 is \(Double(multiplier) * 2.5)"
print(message)

// 打印 "Write an interpolated string in Swift using \(multiplier)."

let Write =  #"Write an interpolated string in Swift using \(multiplier)"#

// 打印 "6 times 7 is 42."
print("6 times 7 is \(6*7)")


let eAcute: Character = "\u{E9}"
let combineEAcute: Character = "\u{65}\u{301}"


let precomposed: Character = "\u{D55C}"                  // 한
let decomposed: Character = "\u{1112}\u{1161}\u{11AB}"   // ᄒ, ᅡ, ᆫ
let decomposed2: Character = "\u{1112}\u{1161}\u{11AB}"   // ᄒ, ᅡ, ᆫ

let unusualMenagerie = "Koala , Snail , Penguin , Dromedary "

// 打印输出“unusualMenagerie has 40 characters”
print("unusualMenagerie has \(unusualMenagerie.count)characters")

var word = "cafe"

word += "\u{301}"    // 拼接一个重音,U+0301

print("the nume of charactor in \(word) is \(word.count)")


let greeting2 = "Guten Tag!"

greeting2[greeting2.startIndex]
greeting2[greeting2.index(before: greeting2.endIndex)]

greeting2[greeting2.index(after: greeting2.startIndex)]

let index = greeting.index(greeting.startIndex, offsetBy: 2)
greeting2[index]

let index2 = greeting2.startIndex
print(index2)


var welcome2 = "hello2"

welcome2.insert("!", at: welcome2.endIndex)

welcome2.insert(contentsOf: "three", at: welcome2.index(before: welcome2.endIndex))
            
print(welcome2)

welcome = "hello there"

welcome.remove(at: welcome.index(before: welcome.endIndex))
print(welcome)

let range = welcome.index(welcome.endIndex, offsetBy: -2)..<welcome.endIndex
welcome.removeSubrange(range)
print(welcome)

// 子字符串

greeting = "Hello, world!"

let index22 = greeting.firstIndex(of: ",") ?? greeting.endIndex
print(index22)
let beginning = greeting[..<index22]
print(beginning)


let newString = String(beginning)

let quotation = "We're a lot alike, you and I."
let someQuotation = "We're a lot alike, you and I."

if quatation == someQuotation {
    print("these two strings are considered equal")
}


let eAcuteQuestion = "Voulez-vous un caf\u{E9}?"
let combinedEAcuteQuestion = "Voulez-vous un caf\u{65}\u{301}?"

if eAcuteQuestion == combinedEAcuteQuestion {
    print("these two strings are considered equal")
}

let romeoAndJuliet = [
    "Act 1 Scene 1: Verona, A public place",
    "Act 1 Scene 2: Capulet's mansion",
    "Act 1 Scene 3: A room in Capulet's mansion",
    "Act 1 Scene 4: A street outside Capulet's mansion",
    "Act 1 Scene 5: The Great Hall in Capulet's mansion",
    "Act 2 Scene 1: Outside Capulet's mansion",
    "Act 2 Scene 2: Capulet's orchard",
    "Act 2 Scene 3: Outside Friar Lawrence's cell",
    "Act 2 Scene 4: A street in Verona",
    "Act 2 Scene 5: Capulet's mansion",
    "Act 2 Scene 6: Friar Lawrence's cell"
]

var act1SceneCount = 0
var act2SceneCount = 0

for scene in romeoAndJuliet {
    if scene.hasPrefix("Act 1") {
        act1SceneCount += 1
    } else if scene.hasPrefix("Act 2") {
        act2SceneCount += 1
    }
}

print("there is has \(act1SceneCount) count \(act2SceneCount) count")

let dogString = "Dog"
for codeUnit in dogString.utf8 {
    print("\(codeUnit)" ,terminator: " ")
}
print("")


for codeUnit in dogString.utf16 {
    print("\(codeUnit)" ,terminator: " ")
}
print("")


for scalar in dogString.unicodeScalars {
    print("\(scalar.value) ", terminator: "")
}
print("")
// 68 111 103 8252 128054

// 集合类型

var someInts: [Int] = []
print("someInts is one has\(someInts.count) 个元素")

someInts.append(3)

someInts = []


var threeDoubles = Array(repeating: 1.0, count: 3)
var threeDoubles2 = Array(repeating: 2.0, count: 3)
let  threeArr = threeDoubles + threeDoubles2
let  threeArr2 = threeDoubles2 + threeDoubles

var someIntsArr: [String] = ["egg","milk"]

var someIntsArr2 = ["egg","aee"]


if someIntsArr2.isEmpty {
    print("The shopping list is  empty")
} else {
    print("The shopping list is not empty")
}

someIntsArr2.append("tags")

var firstItem = someIntsArr2[0]

someIntsArr2[0] = "2333"

print(someIntsArr2)

someIntsArr2[1...2] = ["banana","apple"]
print(someIntsArr2)

someIntsArr2.insert("Mapple Syrup", at: 0)
print(someIntsArr2)

let mappleSyup = someIntsArr2.remove(at: 0)
print(someIntsArr2)

someIntsArr2.removeLast()
print(someIntsArr2)

for item in someIntsArr2 {
    print(item)
}


for (index, value) in someIntsArr2.enumerated() {
    print("item \(index) 的value \(value)")
}

var letters = Set<Character>()
print("letters is of type Set<Character> with \(letters.count) items.")

letters.insert("a")

var favoriteGenres: Set<String> = ["1","2","3"]


var favoriteGenres2: Set = ["11","2","3"]

if favoriteGenres2.isEmpty {
    print("favoriteGenres2 is empty")
}

if favoriteGenres2.contains("11") {
    print("favoriteGenres2 contains 11")
}

for genere in favoriteGenres2 {
    print(genere)
}

for gene in favoriteGenres2.sorted() {
    print(gene)
}

let oddDigits: Set = [1, 3, 5, 7, 9]
let evenDigits: Set = [0, 2, 4, 6, 8]
let singleDigitPrimeNumbers: Set = [2, 3, 5, 7]

oddDigits.union(evenDigits)

print(oddDigits.intersection(evenDigits))


oddDigits.subtracting(singleDigitPrimeNumbers)


let houseAnimals: Set = ["", ""]
let farmAnimals: Set = ["", "", "", "", ""]
let cityAnimals: Set = ["", ""]

houseAnimals.isSubset(of: farmAnimals)
farmAnimals.isSuperset(of: houseAnimals)
farmAnimals.isDisjoint(with: cityAnimals)


// 字典类型简化语法

var namesOfIntegers: [Int: String] = [:]

namesOfIntegers[16] = "sixteen"

var airports: [String: String] = ["YYZ": "Toronto Pearson", "DUB": "Dublin"]

print("has  \(airports.count) count")

airports["LHR"] = "London"

airports["LHR"] = "London Heathrow"



airports["LHR"] = "London Heathrow"

if let oldValue = airports.updateValue("Doubin AirPort", forKey: "DUB") {
    print("The old value for DUB was \(oldValue).")
}
    
if let airportName = airports["DUB"] {
    print("The name of the airport is \(airportName).")
} else {
    print("That airport is not in the airports dictionary.")
}


if let removedValue = airports.removeValue(forKey: "DUB") {
    print("The removed airport's name is \(removedValue)")
} else {
    print("he airports dictionary does not contain a value for DUB.")
}

for (cide, bame) in  airports {
    print("\(cide) \(bame)")
}

for name  in airports.keys {
    print(name)
}

let airportCodes = [String](airports.keys)

//  时间20220531


// 控制流

let  names2 = ["Anna", "Alex", "Brian", "Jack"]
for name in names2 {
    print("hello \(name)")
}


let numberOfLegs = ["spider":8, "ant":7 , "cat":9]
for (name,counts) in numberOfLegs  {
    print("\(name) have \(counts) legs")
}

for index in 1...5 {
    print("\(index) times 5 is \(5 * index)")
}

let base = 3
let power = 10
var answer = 2
for _ in 1...power {
    answer *= base
    print("\(base) to the power of \(power) is answer")
}

print("\(base) to the power of \(power) is answer")

let minutes = 10
for tickMark in 0..<minutes {
    print(tickMark)
}


let hours = 12
let hourInterval = 3
for tickMark in stride(from: 3, through: hours, by: hourInterval) {
    print("tickMark2 = \(tickMark)")
}

let finalSquare = 25
var board = [Int](repeating: 1, count: finalSquare + 1)


var square = 0
var diceRoll = 0

while  square < finalSquare {
    diceRoll += 1
    if diceRoll == 7 {
        diceRoll = 1
    }
    square += diceRoll
    if square < board.count {
        square += board[square]
    }
}

print("Game over")



let someCharacter: Character = "z"

switch someCharacter {
case "a":
    print("The first letter of the alphabet")
case "z":
    print("The last letter of the alphabet")
default:
    print("Some other character")
}


let anotherCharacter:  Character = "a"
switch anotherCharacter {
case "a","A":
    print("the letter A")
default:
    print("Not the letter A")
}


let approximateCount = 62
let countedThings = "moons orbiting Saturn"
let naturalCount: String
switch approximateCount{
case 0:
    naturalCount = "no"
case 1..<5:
    naturalCount = "a few"
case 5..<12:
    naturalCount = "dozens of"
default:
    naturalCount = "many"
}
print("there are \(naturalCount) count")

//元祖

let somePoint = (1,1)
switch somePoint {
case (0,0):
    print("\(somePoint) is at the origin")
case (_,0):
    print("\(somePoint) is at the X")
case (0,_):
    print("\(somePoint) is at the Y")
default:
    print("\(somePoint) is outside of the box")
}

let anotherPoint = (2, 0)
switch anotherPoint {
case (let x, 0):
    print("on the x-axis with an x value of \(x)")
case (0, let y):
    print("on the y-axis with a y value of \(y)")
case let (x, y):
    print("somewhere else at (\(x), \(y))")
}

let yetAnotherPoint = (1, 2)
switch yetAnotherPoint {
case let (x,y) where x == y:
    print("\(x) ,\(y) is on the line x ==y")
case let (x,y) where x == -y:
    print("\(x) ,\(y) is on the line x == -y")
case let (x, y):
    print("\(x) ,\(y)  is just some arbitrary point")
}

let someCharacter2: Character = "e"
switch someCharacter2 {
case "a","e","i","o","u":
    print("\(someCharacter2) is a vowel")
case  "b", "c", "d", "f", "g", "h", "j", "k", "l", "m",
    "n", "p", "q", "r", "s", "t", "v", "w", "x", "y", "z":
    print("\(someCharacter2) is a consonant")
default:
    print("\(someCharacter2) is not a vowel or a consonant")
}



let stillAnotherPoint = (9, 1)
switch stillAnotherPoint {
case (let distance,0) ,(0, let distance) :
    print("On an axis, \(distance) from the origin")
default:
    print("not on an axis")
}

let numberSymbol: Character = "o"  // 简体中文里的数字 3
var possibleIntegerValue: Int?
switch numberSymbol {
case "1", "١", "一", "๑":
    possibleIntegerValue = 1
case "2", "٢", "二", "๒":
    possibleIntegerValue = 2
case "3", "٣", "三", "๓":
    possibleIntegerValue = 3
case "4", "٤", "四", "๔":
    possibleIntegerValue = 4
default:
    break
}

if let integerValue = possibleIntegerValue {
    print("The integer value of \(numberSymbol) is \(integerValue).")
} else {
    print("An integer value could not be found for \(numberSymbol).")
}


let integerToDescribe = 1
var description = "The number \(integerToDescribe) is"
switch integerToDescribe {
case 2, 3, 5, 7, 11, 13, 17, 19:
    description += "a prime number, and also"
default:
    description += "an integer"
}
print(description)


let finalSquare2 = 25
var board1 = [Int](repeating: 0, count: finalSquare2 + 1)
board1[03] = +08; board1[06] = +11; board1[09] = +09; board1[10] = +02
board1[14] = -10; board1[19] = -11; board1[22] = -02; board1[24] = -08
var square1  = 0
var diceRoll1 = 0
square = 3
gameLoop: while square != finalSquare {
    diceRoll += 1
    if diceRoll == 7 { diceRoll = 1 }
    switch square + diceRoll {
    case finalSquare:
        // 骰子数刚好使玩家移动到最终的方格里,游戏结束。
        break gameLoop
    case let newSquare where newSquare > finalSquare:
        // 骰子数将会使玩家的移动超出最后的方格,那么这种移动是不合法的,玩家需要重新掷骰子
        continue gameLoop
    default:
        // 合法移动,做正常的处理
        square += diceRoll
        square += board[square]
    }
}
print("Game over!")


func greet(person:[String:String]) {
    guard let name = person["name"] else {
        return
    }
    print("hello \(name)")
    
    guard let location = person["location"] else {
        print("I hope the weather is nice near you.")
        return
    }
    
    print("I hope the weather is nice in \(location).")
}



greet(person: ["name":"john"])


if #available(iOS 10, macOS 10.12, *) {
    
}else {
    
}


func greet(person: String) -> String {
    let greeting = "hello ," + person + "!"
    return greeting
}

print(greet(person: "Annna"))

print(greet(person: "Brian"))

func greatAgain(person: String) -> String {
    return "hello again ," + person + "!"
}

print(greatAgain(person: "Anna"))



func sayHellowWord() -> String {
    return "hello world"
}

print(sayHellowWord())


func great(person:String, alreadyGreated:Bool) -> String {
    if alreadyGreated {
        return greatAgain(person: person)
    } else {
        return greet(person: person)
    }
}

print(great(person: "Tim", alreadyGreated: false))


func greetNew(person: String) {
    print("hello \(person)")
}
greetNew(person: "test1")

func printAndCount(string1: String) -> Int {
    print(string1)
    return string1.count
}

func printWithOutCounting(string: String) {
    let _ = printAndCount(string1: string)
}

printAndCount(string1: "helloworld")

printWithOutCounting(string: "helloword")


func mixmax(array:[Int]) -> (min: Int, max:Int)? {
    if array.isEmpty {
        return nil
    }
    var currentMin = array[0]
    var currentMax = array[0]
    for value in array[1..<array.count]  {
        if value < currentMin {
            currentMin = value
        } else if value > currentMax {
            currentMax = value
        }
    }
    return (currentMin, currentMax)
}

let bounds = mixmax(array: [1,3, 5, 7,9,19])

if let bounds = mixmax(array: [1,3, 5, 7,9,19]) {
    print("min is \(bounds.min) and max is \(bounds.max)")
}


func greetings(for person: String) -> String {
    "hello ," + person + "!"
}

print(greetings(for: "Dave"))

func anotherGreeting(for person: String) -> String {
    return "hello ," + person + "!"
}

print(anotherGreeting(for: "dave"))


func someFunction(firstParameterName: Int ,secondParameterName:Int) {
    
}

someFunction(firstParameterName: 1, secondParameterName: 2)

func someFunction(argumentLabel parameterName:Int) {
    
}

func greet(person: String ,from hometown: String) -> String {
    return "Hello \(person)!  Glad you could visit from \(hometown)."
}

print(greet(person: "Bill", from: "Cupertino"))


func someFuntion22(_ firstParameterName:Int, secondParameterName:Int) {
    
}
someFuntion22(1, secondParameterName: 2)

func somefuntionValue(parameterWithoutDefault: Int, parameterWithDefault:Int = 12) {
    
}

somefuntionValue(parameterWithoutDefault: 1, parameterWithDefault: 2)

somefuntionValue(parameterWithoutDefault: 4)

// 可变参数

func arithmeticMean(_ numbers: Double...) -> Double {
    var total:Double = 0
    for number in numbers {
        total += number
    }
    return total / Double(numbers.count)
}

arithmeticMean(1,2,3,4,10)

arithmeticMean(3, 8.25, 18.75)


// 输入输出参数

func swapTwoInts(_ a: inout Int, _ b: inout Int) {
    let temporaryA = a
    a  = b
    b  = temporaryA
}

var someInt = 3
var anotherInt = 107
swapTwoInts(&someInt, &anotherInt)//都加了 & 的前缀因为这些量是不能被修改的。当传入的参数作为输入输出参数时,需要在参数名前加 & 符,表示这个值可以被函数修改。

print("someInt is \(someInt) anotherInt is \(anotherInt)")


//函数类型

func addTwoInts(_ a:Int , _ b: Int) -> Int {
    return a + b
}

func multiplyTwoInts(_ a: Int, _ b: Int) -> Int {
    return a * b
}

func printHelloWorld() {
    print("hello word")
}


var mathFunction: (Int, Int) -> Int = multiplyTwoInts

print("Result: \(mathFunction(2,3))")

// 函数类型作为参数类型
func printMathResult(_ mathFunction: (Int , Int) -> Int, _ a: Int, _ b: Int) {
    print("Result:\(mathFunction(a,b))")
}
          
printMathResult(addTwoInts,  3,  5)

//函数类型作为返回类型
func stepForward(_ input: Int) -> Int {
    return input + 1
}

func stepbackWard(_ input: Int) -> Int {
    return input - 1
}

func chooseStepFunction(backward: Bool) -> (Int) -> Int {
    return backward ? stepbackWard : stepForward
}


var currentValue = 3
let moveNearerZero = chooseStepFunction(backward: true)


print("Counting to zero")
while currentValue != 0 {
    print("\(currentValue)")
    currentValue = moveNearerZero(currentValue)
}

print("zero!")


// 嵌套函数

func chooseStepFunction2(backward: Bool) -> (Int) -> Int {
    func stepForward(input:Int) -> Int {return input + 1}
    func stepBaclForward(input: Int) -> Int {return input - 1}
    return backward ? stepForward : stepForward
}

var currentValue2 = -4
let moveNearerToZero = chooseStepFunction2(backward: currentValue > 0)
if currentValue != 0 {
    print("\(currentValue)")
    currentValue =  moveNearerZero(currentValue)
}
print("zero")

// 闭包

let names23 = ["Chris", "Alex", "Ewa", "Barry", "Daniella"]
func backWard23 (_ s1: String , _ s2: String) -> Bool {
    return s1 > s2
}
 
var reversedNames = names23.sorted(by: backWard23)

reversedNames = names23.sorted(by: { (s1: String, s2: String) -> Bool in   return s1 < s2
})

var reversedNames2 = names23.sorted(by:{ s1, s2 in return s1 > s2})

var reversedNames23 = names23.sorted(by:{ s1, s2 in  s1 > s2})

var reversedNames233 = names23.sorted(by:{ $0 > $1})
print("reversedNames233=\(reversedNames233)")

var reversedNames2333 = names23.sorted(by: > )
print("reversedNames2333=\(reversedNames233)")


// 尾随闭包
func someFunctionThatTakesAClosure(closure: () -> Void) {
    // 函数体部分
}

someFunctionThatTakesAClosure ( closure: {
     // 闭包主题部分
  }
)

someFunctionThatTakesAClosure() {
    // 闭包主体部分
}

var reversedNames23334 = names23.sorted{ $0 > $1 }

let digitNames = [
    0: "Zero", 1: "One", 2: "Two",   3: "Three", 4: "Four",
    5: "Five", 6: "Six", 7: "Seven", 8: "Eight", 9: "Nine"
]
let numbers = [16, 58, 510]


// 尾随闭包
let strings = numbers.map {
    (number) -> String in
    var number = number
    var output = ""
    repeat {
        output = digitNames[number%10]! + output
        number /= 10 // 1
    }  while number > 0
    return output
}
print(strings)


// 值捕获
func makeIncrementer(forIncrement amout: Int) -> ()-> Int {
    var runingTotal  = 0
    func incrementer() -> Int {
        runingTotal += amout
        print("runingTotal = \(runingTotal)")
        return runingTotal
    }
    return incrementer
}


let incrementByTen1 = makeIncrementer(forIncrement: 10)
incrementByTen1()
incrementByTen1()

let incrementBySeven = makeIncrementer(forIncrement: 7)
incrementBySeven()

//20220601 10:30
//逃逸闭包
//当一个闭包作为参数传到一个函数中,但是这个闭包在函数返回之后才被执行,我们称该闭包从函数中逃逸。当你定义接受闭包作为参数的函数时,你可以在参数名之前标注 @escaping,用来指明这个闭包是允许“逃逸”出这个函数的。
//一种能使闭包“逃逸”出函数的方法是,将这个闭包保存在一个函数外部定义的变量中。举个例子,很多启动异步操作的函数接受一个闭包参数作为 completion handler。这类函数会在异步操作开始之后立刻返回,但是闭包直到异步操作结束后才会被调用。在这种情况下,闭包需要“逃逸”出函数,因为闭包需要在函数返回之后被调用。例如:
var completionHanders: [() -> Void] = []
func someFunctionWithEscapingClosure(completionHandler: @escaping () -> Void )
{
    completionHanders.append(completionHandler)
}
//
//将一个闭包标记为 @escaping 意味着你必须在闭包中显式地引用 self。比如说,在下面的代码中,传递到 someFunctionWithEscapingClosure(_:) 中的闭包是一个逃逸闭包,这意味着它需要显式地引用 self。相对的,传递到 someFunctionWithNonescapingClosure(_:) 中的闭包是一个非逃逸闭包,这意味着它可以隐式引用 self。


func someFunctionWithNonescapingClosure(closure: () -> Void) {
    closure()
}

class SomeClass {
    var x = 10
    func doSomeThing() {
        someFunctionWithEscapingClosure {
            self.x = 100 //将一个闭包标记为 @escaping 意味着你必须在闭包中显式地引用 self
        }
        
        someFunctionWithNonescapingClosure {
            x = 200
        }
    }
}


let instance = SomeClass()
instance.doSomeThing()
print(instance.x) // 非逃逸闭包

completionHanders.first?()
print(instance.x) // 逃逸闭包 . 被调用才生效


//自动闭包
/*
自动闭包
自动闭包是一种自动创建的闭包,用于包装传递给函数作为参数的表达式。这种闭包不接受任何参数,当它被调用的时候,会返回被包装在其中的表达式的值。这种便利语法让你能够省略闭包的花括号,用一个普通的表达式来代替显式的闭包。
我们经常会调用采用自动闭包的函数,但是很少去实现这样的函数。举个例子来说,assert(condition:message:file:line:) 函数接受自动闭包作为它的 condition 参数和 message 参数;它的 condition 参数仅会在 debug 模式下被求值,它的 message 参数仅当 condition 参数为 false 时被计算求值。
自动闭包让你能够延迟求值,因为直到你调用这个闭包,代码段才会被执行。延迟求值对于那些有副作用(Side Effect)和高计算成本的代码来说是很有益处的,因为它使得你能控制代码的执行时机。下面的代码展示了闭包如何延时求值。
 */

var customersInLine = ["Chris", "Alex", "Ewa", "Barry", "Daniella", "Daniella2", "Daniella3", "Daniella4"]
print("customersInLine=\(customersInLine.count))")

let customerProvider = {customersInLine.remove(at: 0)}
print(customersInLine.count)

print("Now serving \(customerProvider())!")
print(customersInLine.count)


func serve(customer customerProvider: () -> String) {
    print("Now serving \(customerProvider())!")
}

serve(customer: {customersInLine.remove(at: 0)})
func serve(customer customerProvider: @autoclosure () -> String) {
    print("Now serving \(customerProvider())!")
}

print(serve(customer: customersInLine.remove(at: 0)))

//如果你想让一个自动闭包可以“逃逸”,则应该同时使用 @autoclosure 和 @escaping 属性。@escaping 属性的讲解见上面的 逃逸闭包。

var customerProviders: [() -> String] = []

func collectCustomerProviders(_ customerProvider:@autoclosure @escaping () -> String) {
    customerProviders.append(customerProvider)
}
collectCustomerProviders(customersInLine.remove(at: 0))
collectCustomerProviders(customersInLine.remove(at: 0))

print("Collected \(customerProviders.count) closures.")

for customerProvider in customerProviders {
    print("Now serving \(customerProvider())")
}

上一篇 下一篇

猜你喜欢

热点阅读