Aop神器Aspects在OC下使用簡(jiǎn)單。但是在swift下能否正常使用呢?
答案是肯定的。但是也有些限制,aspects利用objc runtime的方法來(lái)實(shí)現(xiàn)的,所以在swift下,只能觀察swift下的NSObject(帶有@objc關(guān)鍵字的對(duì)象)。這里還是建議新建對(duì)象的時(shí)候直接繼承NSObject。這樣就不用自己標(biāo)注@objc關(guān)鍵字。
在 Swift 類(lèi)型文件中,我們可以將需要暴露給 Objective-C 使用的任何地方 (包括類(lèi),屬性和方法等) 的聲明前面加上 @objc 修飾符。注意這個(gè)步驟只需要對(duì)那些不是繼承自 NSObject 的類(lèi)型進(jìn)行,如果你用 Swift 寫(xiě)的 class 是繼承自 NSObject 的話,Swift 會(huì)默認(rèn)自動(dòng)為所有的非 private 的類(lèi)和成員加上 @objc
來(lái)自Swift和Objective-C如何兼顧?且看@objc和Dynamic
接著來(lái)看Aspects的一個(gè)方法
- (id<AspectToken>)aspect_hookSelector:(SEL)selector withOptions:(AspectOptions)options
usingBlock:(id)block
error:(NSError **)error;
為了對(duì)比這里放一下UIView的一個(gè)方法
+ (void)animateWithDuration:(NSTimeInterval)duration animations:(void (^)(void))animations
注意block的類(lèi)型,UIView的方法中已經(jīng)直接聲明好參數(shù)和返回值。Aspects為了靈活則直接聲明為了一個(gè)id對(duì)象。id對(duì)應(yīng)為swift中的AnyObject對(duì)象。所以在swift中為了正常使用這個(gè)方法,需要將閉包轉(zhuǎn)換為AnyObject。代碼如下:
let wrappedBlock:@convention(block) (AspectInfo)-> Void = { aspectInfo in
// 你的代碼
}
let wrappedObject: AnyObject = unsafeBitCast(wrappedBlock, to: AnyObject.self)
注意這里是unsafe的,在swift2.0后不安全的代碼需要做異常處理。否則會(huì)有
Call can throw, but it is not marked with 'try' and the error is not handled
調(diào)用代碼如下:
do {
try self.aspect_hookSelector(Selector("groupMenuTouch:"), withOptions: AspectOptions.PositionBefore, usingBlock: wrappedObject)
}catch{
print(error)
}
相關(guān)鏈接:
Swift closure as AnyObject
歡迎關(guān)注我的微博:@沒(méi)故事的卓同學(xué)