Как объявить блок с аргументами в 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'")