如何在swift下调用Aspects:将闭包封装成AnyObject

Aop神器Aspects在OC下使用简单。但是在swift下能否正常使用呢?
答案是肯定的。但是也有些限制,aspects利用objc runtime的方法来实现的,所以在swift下,只能观察swift下的NSObject(带有@objc关键字的对象)。这里还是建议新建对象的时候直接继承NSObject。这样就不用自己标注@objc关键字。

在 Swift 类型文件中,我们可以将需要暴露给 Objective-C 使用的任何地方 (包括类,属性和方法等) 的声明前面加上 @objc 修饰符。注意这个步骤只需要对那些不是继承自 NSObject 的类型进行,如果你用 Swift 写的 class 是继承自 NSObject 的话,Swift 会默认自动为所有的非 private 的类和成员加上 @objc

来自Swift和Objective-C如何兼顾?且看@objc和Dynamic

接着来看Aspects的一个方法

- (id)aspect_hookSelector:(SEL)selector withOptions:(AspectOptions)options
                       usingBlock:(id)block
                            error:(NSError **)error;

为了对比这里放一下UIView的一个方法

+ (void)animateWithDuration:(NSTimeInterval)duration animations:(void (^)(void))animations

注意block的类型,UIView的方法中已经直接声明好参数和返回值。Aspects为了灵活则直接声明为了一个id对象。id对应为swift中的AnyObject对象。所以在swift中为了正常使用这个方法,需要将闭包转换为AnyObject。代码如下:

      let wrappedBlock:@convention(block) (AspectInfo)-> Void = { aspectInfo in
            // 你的代码
        }
        let wrappedObject: AnyObject = unsafeBitCast(wrappedBlock, to: AnyObject.self)

注意这里是unsafe的,在swift2.0后不安全的代码需要做异常处理。否则会有

Call can throw, but it is not marked with 'try' and the error is not handled

调用代码如下:

        do {
           try self.aspect_hookSelector(Selector("groupMenuTouch:"), withOptions: AspectOptions.PositionBefore, usingBlock: wrappedObject)
        }catch{
            print(error)
        }

相关链接:
Swift closure as AnyObject

欢迎关注我的微博:@没故事的卓同学

你可能感兴趣的:(如何在swift下调用Aspects:将闭包封装成AnyObject)