【问题标题】:Swift blocks not workingSwift 块不起作用
【发布时间】:2014-08-27 01:20:37
【问题描述】:

我一直在试图弄清楚如何在 swift 中使用 JavaScriptCore。我遇到了问题,但是当我必须将块作为参数处理时,似乎块立即运行并且参数获取块的返回值。我做错了什么?

工作目标 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'"

我在这里遗漏了什么,还是这只是 Swift 中的一个错误?

编辑:

我现在也尝试了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)

【问题讨论】:

标签: swift block javascriptcore


【解决方案1】:

这与 Swift 如何实现闭包有关。您需要使用@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

【讨论】:

  • 这就像一个魅力!太感谢了。唯一的事情,显然ctx["test"] = 不起作用。但是ctx.setObject(obj, forKeyedSubscript:"test") 确实如此。再次感谢您,我可以在 22 小时内为您提供赏金。
  • 在 Beta6 上,reinterpretCast(block) as AnyObject 已被弃用,取而代之的是 unsafeBitCast(block, AnyObject)
  • 其实应该是unsafeBitCast(block, AnyObject.self)
  • 这一行 var ctx["test"] = reinterpretCast(block) as AnyObject 应该是 ctx.setObject(unsafeBitCast(block, AnyObject.self), forKeyedSubscript: "test"),在 Xcode 版本 6.1.1 (6A2008a) 上测试。
  • 使用最新的 Swift 2 语法,块应该定义为@convention(block) (String) -&gt; Void
【解决方案2】:

我有一个工作演示:

这里是实现块注册的部分:

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 代码和桥接头才能使其工作。有关详细信息,请参阅存储库。

【讨论】:

    猜你喜欢
    • 2016-08-26
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多