一个简洁高效的用于使iOS支持AOP面向切面编程的库.它能够帮助你在不改变一个类或类实例的代码的前提下,有效更改类的行为.比iOS传统的 AOP方法,更加简单高效.支持在方法执行的前/后或替代原方法执行.曾经是 PSPDFKit 的一部分,PSPDFKit,在Dropbox和Evernote中都有应用,如今单独单独开源出来给你们使用.ios
项目主页: Aspectsgit
最新实例:点击下载github
注: AOP是一种彻底不一样于OOP的设计模式.更多信息,能够参考这里: AOP 百度百科编程
ARC设计模式
iOS 7 + 或 OS X 10.7 +安全
pod "Aspects"
把文件 Aspects.h/m
拖到工程中便可.服务器
Aspects 用于支持AOP(面向切面编程)模式,用于部分解决OOP(面向对象)模式没法解决的特定问题.具体指的是那些在多个方法有交叉,没法或很难被有效归类的操做,好比:app
不论什么时候用户经过客户端获取服务器端数据,权限检查老是必须的.
不论什么时候用户和市场交互,总应该更具用户的操做提供相应地购买参考或相关商品.
全部须要日志记录的操做.性能
Aspects 给 NSObject 扩展了下面的方法:测试
/// 为一个指定的类的某个方法执行前/替换/后,添加一段代码块.对这个类的全部对象都会起做用. /// /// @param block 方法被添加钩子时,Aspectes会拷贝方法的签名信息. /// 第一个参数将会是 `id<AspectInfo>`,余下的参数是此被调用的方法的参数. /// 这些参数是可选的,并将被用于传递给block代码块对应位置的参数. /// 你甚至使用一个没有任何参数或只有一个`id<AspectInfo>`参数的block代码块. /// /// @注意 不支持给静态方法添加钩子. /// @return 返回一个惟一值,用于取消此钩子. + (id<AspectToken>)aspect_hookSelector:(SEL)selector withOptions:(AspectOptions)options usingBlock:(id)block error:(NSError **)error; /// 为一个指定的对象的某个方法执行前/替换/后,添加一段代码块.只做用于当前对象. - (id<AspectToken>)aspect_hookSelector:(SEL)selector withOptions:(AspectOptions)options usingBlock:(id)block error:(NSError **)error; - (id<AspectToken>)aspect_hookSelector:(SEL)selector withOptions:(AspectOptions)options usingBlock:(id)block error:(NSError **)error; /// 撤销一个Aspect 钩子. /// @return YES 撤销成功, 不然返回 NO. id<AspectToken> aspect = ...; [aspect remove];
全部的调用,都会是线程安全的.Aspects 使用了Objective-C 的消息转发机会,会有必定的性能消耗.全部对于过于频繁的调用,不建议使用 Aspects.Aspects更适用于视图/控制器相关的等每秒调用不超过1000次的代码.
能够在调试应用时,使用Aspects动态添加日志记录功能.
[UIViewController aspect_hookSelector:@selector(viewWillAppear:) withOptions:AspectPositionAfter usingBlock:^(id<AspectInfo> aspectInfo, BOOL animated) { NSLog(@"控制器 %@ 将要显示: %tu", aspectInfo.instance, animated); } error:NULL];
使用它,分析功能的设置会很简单:
https://github.com/orta/ARAnalytics
你能够在你的测试用例中用它来检查某个方法是否被真正调用(当涉及到继承或类目扩展时,很容易发生某个父类/子类方法未按预期调用的状况):
- (void)testExample { TestClass *testClass = [TestClass new]; TestClass *testClass2 = [TestClass new]; __block BOOL testCallCalled = NO; [testClass aspect_hookSelector:@selector(testCall) withOptions:AspectPositionAfter usingBlock:^{ testCallCalled = YES; } error:NULL]; [testClass2 testCallAndExecuteBlock:^{ [testClass testCall]; } error:NULL]; XCTAssertTrue(testCallCalled, @"调用testCallAndExecuteBlock 必须调用 testCall"); }
它对调试应用真的会提供很大的做用.这里我想要知道究竟什么时候轻击手势的状态发生变化(若是是某个你自定义的手势的子类,你能够重写setState:方法来达到相似的效果;但这里的真正目的是,捕捉全部的各种控件的轻击手势,以准确分析缘由):
[_singleTapGesture aspect_hookSelector:@selector(setState:) withOptions:AspectPositionAfter usingBlock:^(id<AspectInfo> aspectInfo) { NSLog(@"%@: %@", aspectInfo.instance, aspectInfo.arguments); } error:NULL];
下面是一个你监测一个模态显示的控制器什么时候消失的示例.一般,你也能够写一个子类,来实现类似的效果,但使用 Aspects 能够有效减少你的代码量:
@implementation UIViewController (DismissActionHook) // Will add a dismiss action once the controller gets dismissed. - (void)pspdf_addWillDismissAction:(void (^)(void))action { PSPDFAssert(action != NULL); [self aspect_hookSelector:@selector(viewWillDisappear:) withOptions:AspectPositionAfter usingBlock:^(id<AspectInfo> aspectInfo) { if ([aspectInfo.instance isBeingDismissed]) { action(); } } error:NULL]; } @end
Aspectes 会自动标记本身,全部很容易在调用栈中查看某个方法是否已经调用:
你可使用 NSInvocation 对象类自定义返回值:
[PSPDFDrawView aspect_hookSelector:@selector(shouldProcessTouches:withEvent:) withOptions:AspectPositionInstead usingBlock:^(id<AspectInfo> info, NSSet *touches, UIEvent *event) { // 调用方法原来的实现. BOOL processTouches; NSInvocation *invocation = info.originalInvocation; [invocation invoke]; [invocation getReturnValue:&processTouches]; if (processTouches) { processTouches = pspdf_stylusShouldProcessTouches(touches, event); [invocation setReturnValue:&processTouches]; } } error:NULL];
当应用于某个类时(使用类方法添加钩子),不能同时hook父类和子类的同一个方法;不然会引发循环调用问题.可是,当应用于某个类的示例时(使用实例方法添加钩子),不受此限制.
使用KVO时,最好在 aspect_hookSelector:
调用以后添加观察者;不然可能会引发崩溃.