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<AspectToken>)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
欢迎关注我的微博:@没故事的卓同学