BlocksKit初见:一个支持将delegate转换成block的Cocoa库

简介

项目主页:https://github.com/zwaldowski/BlocksKit

BlocksKit 是一个开源的框架,对 Cocoa 进行了扩展,将许多需要通过 delegate 调用的方法转换成了 block。在很多情况下,blocks 比 delegate 要方便简单,因为 block 是紧凑的,可以使代码简洁,提高代码可读性,另外 block 还可以进行异步处理。使用 block 要注意避免循环引用。

目录结构

BlocksKit 的所有方法都以bk_开头,这样可以方便地列出所有 BlocksKit 的所有方法。BlocksKit 主要目录结构

Core:存放 Foundation 相关的 Block category,如 NSObject、NSTimer、NSarray、NSDictionary、NSSet、NSIndexSet、NSMutableArray等

DynamicDelegate:动态代理(消息转发机制)

UIKit:扩展了 UIAlertView,UIActionView,UIButton 等

最常用的是 UIKit Category,它为 UIAlertView,UIActionSheet,UIButton,UITapGestureRecognizer 等提供了 blocks。

用法实例

UIAlertView 和 UIActionSheet 用法示例:

10UIAlertView*alertView=[[UIAlertViewalloc]bk_initWithTitle:@"提示"message:@"提示信息"];

[alertViewbk_setCancelButtonWithTitle:@"取消"handler:nil];

[alertViewbk_addButtonWithTitle:@"确定"handler:nil];

[alertViewbk_setDidDismissBlock:^(UIAlertView*alert,NSIntegerindex){

if(index==1){

NSLog(@"%ld clicked",index);

}

}];

[alertViewshow];

5[[UIActionSheetbk_actionSheetCustomWithTitle:nilbuttonTitles:@[@"查看",@"退出"]destructiveTitle:nilcancelTitle:@"取消"andDidDismissBlock:^(UIActionSheet*sheet,NSIntegerindex){

}]showInView:self.view];

UIButton 和 UITapGestureRecognizer 用法示例:

6UIButton*button=[[UIButtonalloc]init];

[buttonbk_addEventHandler:^(idsender){

}forControlEvents:UIControlEventTouchUpInside];

6UITapGestureRecognizer*tapGestureRecognizer=[UITapGestureRecognizerbk_recognizerWithHandler:^(UIGestureRecognizer*sender,UIGestureRecognizerStatestate,CGPointlocation){

if(state==UIGestureRecognizerStateRecognized){

...

}

}];

UIButton 和 UIGesture 将 target-action 转换成 block,实现较简单:

-(id)bk_initWithHandler:(void(^)(UIGestureRecognizer*sender,UIGestureRecognizerStatestate,CGPointlocation))blockdelay:(NSTimeInterval)delay

{

self=[selfinitWithTarget:selfaction:@selector(bk_handleAction:)];

if(!self)returnnil;

self.bk_handler=block;

self.bk_handlerDelay=delay;

returnself;

}

-(void)bk_handleAction:(UIGestureRecognizer*)recognizer

{

void(^handler)(UIGestureRecognizer*sender,UIGestureRecognizerStatestate,CGPointlocation)=recognizer.bk_handler;

if(!handler)return;

...

if(!delay){

block();

return;

}

...

}

delegate 转换成 block 实际上使用了消息转发机制,是 BlocksKit 源码中最难理解的部分。

原理分析: 消息转发机制

当一个对象收到它没实现的消息的时候,通常会发生如下的情况。

调用+(BOOL)resolveInstanceMethod:(SEL)aSEL,如果对象在这里动态添加了selector 的实现方法,则消息转发结束,否则执行步骤2

调用- (id)forwardingTargetForSelector:(SEL)aSelector,在这里你可以将消息转发给其他对象,如果实现则消息转发结束,否则执行步骤3

执行完整的消息转发机制,调用-(void)forwardInvocation:(NSInvocation *)invocation在这一步,你可以修改消息的任何内容,包括目标(target),selector,参数。如果没有实现在这里还未实现转发则程序将抛出异常。

原理实例分析

BlocksKit 动态代理实现方式是最后一步,即-(void)forwardInvocation:(NSInvocation *)invocation,使得动态代理能够接受任意消息。

以UIAlertView为例,UIAlertView在运行时动态关联了A2DynamicUIAlertViewDelegate

@implementationUIAlertView(BlocksKit)

@dynamicbk_willShowBlock,bk_didShowBlock,bk_willDismissBlock,bk_didDismissBlock,bk_shouldEnableFirstOtherButtonBlock;

+(void)load

{

@autoreleasepool{

[selfbk_registerDynamicDelegate];

[selfbk_linkDelegateMethods:@{

@"bk_willShowBlock":@"willPresentAlertView:",

@"bk_didShowBlock":@"didPresentAlertView:",

@"bk_willDismissBlock":@"alertView:willDismissWithButtonIndex:",

@"bk_didDismissBlock":@"alertView:didDismissWithButtonIndex:",

@"bk_shouldEnableFirstOtherButtonBlock":@"alertViewShouldEnableFirstOtherButton:"

}];

}

}

A2DynamicUIAlertViewDelegate 是 A2DynamicDelegate 的子类,并实现了UIAlertViewDelegate 的方法

代理消息的转发由 A2DynamicDelegate 完成

-(void)forwardInvocation:(NSInvocation*)outerInv

{

SELselector=outerInv.selector;

A2BlockInvocation*innerInv=nil;

if((innerInv=[self.invocationsBySelectorsbk_objectForSelector:selector])){

[innerInvinvokeWithInvocation:outerInv];

}elseif([self.realDelegaterespondsToSelector:selector]){

[outerInvinvokeWithTarget:self.realDelegate];

}

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,789评论 0 9
  • 1、设置UILabel行间距 NSMutableAttributedString* attrString = [[...
    十年一品温如言1008阅读 1,731评论 0 3
  • { 11、核心动画 需要签协议,但是系统帮签好 一、CABasicAnimation 1、创建基础动画对象 CAB...
    CYC666阅读 1,591评论 2 4
  • iOS开发中,页面传值是很常见的,但是页面传值你究竟知道多少呢?笔者这篇文章就是给大家介绍一下页面传值的具体方式,...
    蒲公英少年阅读 2,186评论 10 44
  • 每天都在相同的路上行走,不知道远方。 对小v而言,生活无非就是简简单单,安安静静,平平淡淡。或许不是生活杀死了热情...
    雀斑少女v阅读 346评论 0 0