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!"