Блоки Swift не работают
Я пытался понять, как использовать JavaScriptCore в swift. Я сталкиваюсь с проблемами, однако, когда мне приходится иметь дело с блоками в качестве аргументов, кажется, что блок запускается немедленно, и аргументы получают возвращаемое значение блока. Что я делаю не так?
Рабочая цель C-код:
JSContext* context = [[JSContext alloc] initWithVirtualMachine:[[JSVirtualMachine alloc] init]];
context[@"test"] = ^(NSString *string) {
//code
};
Что я пробовал:
1:
var ctx = JSContext(virtualMachine:JSVirtualMachine())
var ctx["test"] = {(string:NSString)->() in /*code*/ }
//Gives me "'JSContext' does not have a member named 'subscript'"
2:
var ctx = JSContext(virtualMachine:JSVirtualMachine())
let n: (string: String)->() = {string in /*code*/}
ctx.setObject(n, forKeyedSubscript:"test")
//Gives me "Type '(x: String) -> () does not conform to protocol 'AnyObject'"
3:
var ctx = JSContext(virtualMachine:JSVirtualMachine())
let n: (string: String)->() = {string in /*code*/}
ctx.setObject(n as AnyObject, forKeyedSubscript:"test")
//Gives me "Cannot downcast from '(string: String) -> () to non-@objc protocol type 'AnyObject'"
Я что-то упустил, или это просто ошибка Свифт?
Правка:
Теперь я также попробовал предложения из Cast closures / blocks
class Block<T> {
let f : T
init (_ f: T) { self.f = f }
}
И затем
ctx.setObject(Block<()->Void> {
/*code*/
}, forKeyedSubscript: "test")
Это решение позволяет мне компилировать, но я получаю ошибку времени выполнения:
Thread 1: EXC_BREAKPOINT (code=EXC_I386_BPT, subcode=0x0)
2 ответа:
Это как-то связано с тем, как быстро реализовать закрытие. Вы должны использовать
@convention(block)
, чтобы отметить, что закрытие является блоком ObjC. ИспользуйтеunsafeBitCast
, чтобы заставить бросить егоvar block : @convention(block) (NSString!) -> Void = { (string : NSString!) -> Void in println("test") } ctx.setObject(unsafeBitCast(block, AnyObject.self), forKeyedSubscript: "test")
От REPL
swift Welcome to Swift! Type :help for assistance. 1> import Foundation 2> var block : @convention(block)(NSString!) -> Void = {(string : NSString!) -> Void in println("test")} block: @convention(block)(NSString!) -> Void = 3> var obj: AnyObject = reinterpretCast(block) as AnyObject obj: __NSMallocBlock__ = {} // familiar block type
У меня есть рабочая демо-версия по адресу:
А вот та часть, которая реализует регистрацию блоков:
typealias ID = AnyObject! extension JSContext { func fetch(key:NSString)->JSValue { return getJSVinJSC(self, key) } func store(key:NSString, _ val:ID) { setJSVinJSC(self, key, val) } func store(key:NSString, _ blk:()->ID) { setB0JSVinJSC(self, key, blk) } func store(key:NSString, _ blk:(ID)->ID) { setB1JSVinJSC(self, key, blk) } func store(key:NSString, _ blk:(ID,ID)->ID) { setB2JSVinJSC(self, key, blk) } }
Вам нужен очень маленький код objc и заголовок моста, чтобы заставить его работать. Подробности смотрите в хранилище.