Как объявить блок с аргументами в Swift?

трудно понять, как правильно объявлять/использовать блоки с swift. Каким будет swift-эквивалент следующего кода?

спасибо.

^(PFUser *user, NSError *error) {
if (!user) {
    NSLog(@"Uh oh. The user cancelled the Facebook login.");
} else if (user.isNew) {
    NSLog(@"User signed up and logged in through Facebook!");
} else {
    NSLog(@"User logged in through Facebook!");
}

6 ответов


эквивалентом блоков Objective-C являются быстрые замыкания, поэтому он будет выглядеть следующим образом

{ (user: PFUser, error: NSError) in
  if (!user) {
    println("Uh oh. The user cancelled the Facebook login.");
  } else if (user.isNew) {
    println("User signed up and logged in through Facebook!");
  } else {
    println("User logged in through Facebook!");
  }
}

у вас есть много способов, предлагаемых для передачи блока, эквивалентного функции в Swift.

Я нашел три.

чтобы понять это, я предлагаю вам проверить на площадке этот маленький кусочек кода.

func test(function:String -> String) -> String
{
    return function("test")
}

func funcStyle(s:String) -> String
{
    return "FUNC__" + s + "__FUNC"
}
let resultFunc = test(funcStyle)

let blockStyle:(String) -> String = {s in return "BLOCK__" + s + "__BLOCK"}
let resultBlock = test(blockStyle)

let resultAnon = test({(s:String) -> String in return "ANON_" + s + "__ANON" })


println(resultFunc)
println(resultBlock)
println(resultAnon)

Update: есть 2 особых случая для анонимной функции.

во-первых, сигнатура функции может быть выведена, поэтому вам не нужно переписывать ее.

let resultShortAnon = test({return "ANON_" +  + "__ANON" })

второй специальный случай работает только в том случае, если блок является последним аргумент, он называется трейлинг-закрытие

вот пример (объединенный с выводимой подписью, чтобы показать Swift power)

let resultTrailingClosure = test { return "TRAILCLOS_" +  + "__TRAILCLOS" }

и наконец:

используя всю эту силу, что я бы сделал, это смешивание трейлинг-закрытия и вывода типа (с именованием для удобочитаемости)

PFFacebookUtils.logInWithPermissions(permissions) {
    user, error in
    if (!user) {
        println("Uh oh. The user cancelled the Facebook login.")
    } else if (user.isNew) {
        println("User signed up and logged in through Facebook!")
    } else {
        println("User logged in through Facebook!")
    }
}

IMO это красивее, чем в ObjC


посмотрите, работает ли это для вас. Это безумие-пытаться узнать это на второй день.

let afterSignInAttempt: (PFUser?, NSError) -> Void = { user, error in
    if(!user){
      NSLog("Uh oh.")
    } else {
      user.isNew ? NSLog("Signed up") : NSLog("User Logged in")
    }
}

критично, если user может быть nil затем должно быть объявлено необязательным. Таким образом:

{ (user: PFUser?, error: NSError) -> {} in
         if (nil == user) ...
    }

отметить, что тип user включает в себя ? указывая, что user это необязательный аргумент (либо nil или типа PFUser).

другие ответы, которые не используют необязательный, даже не компилировать.


Если вы хотите сохранить блок в переменной и вызвать его позже, проверьте это ответ


// определить его

class IDDBlockTime {
    // return time elapsed in milliseconds
    //
    static func timeSpent(_ block: (Void) -> Void) -> TimeInterval {
        let methodStart = Date()

        block()
        return Date().timeIntervalSince(methodStart) * 1000.0
    }
}

// использовать

    let timeSpent = IDDBlockTime.timeSpent {
        // lines of code you want to measure
        //
        self.doSomethig()
    }

    print("timeSpent: '\(timeSpent) ms'")