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
歡迎關注我的微博:@沒故事的卓同學