Swift - магазин закрытий в словаре
Можно ли хранить затворы в словарях (как мы могли бы хранить блоки ObjC в словарях)? Пример:
data = [String:AnyObject]()
data!["so:c0.onSelection"] = {() in
Debug.log(.Debug, message: "Hello, World!")
}
Ответы
Ответ 1
Вы можете, но с некоторыми ограничениями. Прежде всего, типы функций не наследуются от AnyObject и не имеют общего базового класса. У вас может быть словарь [String: () -> Void]
и [String: (String) -> Int]
, но они не могут быть сохранены в одном словаре.
Мне также пришлось использовать typealias для определения словаря, чтобы swift правильно разбирался. Вот пример, основанный на вашем фрагменте.
typealias myClosure = () -> Void
var data: [String: myClosure]? = [String: myClosure]()
data!["so:c0.onSelection"] = {() -> Void in
Debug.log(.Debug, message: "Hello, World!")
}
Ответ 2
У меня другой подход
Я создаю класс "держатель", чтобы удерживать ваши замыкания примерно так:
typealias SocialDownloadImageClosure = (image : UIImage?, error: NSError?) -> ()
typealias SocialDownloadInformationClosure = (userInfo : NSDictionary?, error: NSError?) -> ()
private class ClosureHolder
{
let imageClosure:SocialDownloadImageClosure?
let infoClosure:SocialDownloadInformationClosure?
init(infoClosure:SocialDownloadInformationClosure)
{
self.infoClosure = infoClosure
}
init(imageClosure:SocialDownloadImageClosure)
{
self.imageClosure = imageClosure
}
}
тогда я делаю словарь следующим образом:
var requests = Dictionary<String,ClosureHolder>()
Теперь, чтобы добавить замыкание в словарь, просто выполните следующее:
self.requests["so:c0.onSelection"] = ClosureHolder(completionHandler)
Ответ 3
Коннор прав, я пытался много способов хранить переменные и замыкания в одном словаре, но я потерпел неудачу и не смог разобрать его, быстрый декомпилятор выбросит ошибку:
"Command failed due to signal: Segmentation fault: 11" (the hell is it?!)
Например:
//This won't work
var params:[String: Any] = ["x":100, "onFoundX": {println("I found X!")}]
if var onFoundX: (()->Void) = params["onFoundX"] as? (()->Void) {
onFoundX()
}
//This should work by separate into 2 dictionaries and declare the "typealias" obviously
var params:[String: Any] = ["x":"100"}]
var events:[String: (()->Void)] = ["onFoundX": {println("I found X!")]
if var onFoundX: (()->Void) = events["onFoundX"] as? (()->Void) {
onFoundX() // "I found X!"
}
if var x = events["x"] as? String {
println(x) // 100
}
Я надеюсь, что Свифт позволит это произойти в будущем.
Ура!
Ответ 4
Этот простой пример помог мне понять немного больше:
//Init dictionary with types (i.e. String type for key, Closure type for value):
var myDictionary: [String: ()->(String)] = [:]
//Make a closure that matches the closure signature above and assign to variable (i.e. no parameter and returns a String):
let sayHello: () -> (String) = {
return "Hello!"
}
//Add closure to dictionary with key:
myDictionary["myFunc"] = sayHello
//Access closure by known key and call it:
myDictionary["myFunc"]!() //"Hello!"