AnsycDisplayKit是关注的人比较少的库之一,这是因为这是个很重量级的库,它基本重写了UIKit,使用它基本上就等同于放弃原来的UIView和UILayer的方案,还有个原因是很少有界面复杂到像Facebook那样对体验要求那么高。但这些问题都不影响我们探究它内部的机制,毕竟这是个Facebook内部使用的库。
AnsycDisplayKit 的下载地址 https://github.com/facebookarchive/AsyncDisplayKit
正如github上所说,AsyncDisplayKit已经重新命名为Texture ,究其原因笔者猜测是因为作者(Scott Goodson)的离职。他曾经就职于Facebook以及Instagram等公司,并在这里大致介绍了AsyncDisplayKit 的概况 :
Scott Goodson - Behind AsyncDisplayKit
这个库太庞大了,以至于我们不可能在一篇文章中描述完全,因此,笔者会做个系列博客和大家讨论这个库。今天我们讲第2篇:关于NSProxy
本文的知识点:
- iOS中的消息转发机制
- 多继承
- AOP
众所周知,iOS的方法调用会经历三个阶段:
消息转发分为三大阶段
- 第一阶段先征询消息接收者所属的类,看其是否能动态添加方法,以处理当前这个无法响应的 selector,这叫做 动态方法解析(dynamic method resolution)。如果运行期系统(runtime system) 第一阶段执行结束,接收者就无法再以动态新增方法的手段来响应消息,进入第二阶段。
- 第二阶段看看有没有其他对象(备援接收者,replacement receiver)能处理此消息。如果有,运行期系统会把消息转发给那个对象,转发过程结束;如果没有,则启动完整的消息转发机制。
- 第三阶段 完整的消息转发机制。运行期系统会把与消息有关的全部细节都封装到 NSInvocation 对象中,再给接收者最后一次机会,令其设法解决当前还未处理的消息。
我们就在第二阶段进行插入代理者的操作,代码如下所示(其中Animal是继承自NSObject的类)
Cat.h
@interface Cat : Animal
-(void)say;
@end
Cat.m
@implementation Cat
-(void)say
{
NSLog(@"miao");
}
@end
看以上代码,我们可以发现,“猫”拥有说话的能力。
Pet.h
@interface Pet : NSObject
@property (nonatomic, strong) NSObject *intercetper;
-(void) play;
@end
Pet.m
@implementation Pet
-(void) play{
NSLog(@"pay");
}
-(void)forwardInvocation:(NSInvocation *)invocation
{
[invocation setTarget:self.intercetper];
[invocation invoke];
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel
{
NSMethodSignature *signature = nil;
if ([self.intercetper methodSignatureForSelector:sel]) {
signature = [self.intercetper methodSignatureForSelector:sel];
}else{
signature = [self methodSignatureForSelector:sel];
}
return signature;
}
@end
由Pet的代码可知Pet只有Play的能力,没有say的能力。
最后我们在ViewController中调用如下代码:
Cat *cat = [[Cat alloc] init];
Pet *pet = [[Pet alloc] init];
pet.intercetper = cat;
[pet performSelector:@selector(say) withObject:nil];
我们会发现Pet也拥有了说话的能力。
这是因为Pet在调用say方法的时候发现找不到,于是它去调用了methodSignatureForSelector
方法。
这就是第二阶段的含义。
我们仍然以上一篇的Demo为例,我们一一分析。
- (void)viewDidLoad
{
[super viewDidLoad];
self.title = @"Image Categories";
self.node.delegate = self;
self.node.dataSource = self;
}
这其实就是设置了tableview的delegate以及dataSource。并且还加入了自己的一些方法。这里我们以设置Delegate为例进行讲解。
- (void)setDelegate:(id <ASTableDelegate>)delegate
{
if ([self pendingState]) {
_pendingState.delegate = delegate;
} else {
//这里获取TableView
ASTableView *view = self.view;
//这里设置Delegate
ASPerformBlockOnMainThread(^{
view.asyncDelegate = delegate;
});
}
}
如图,view.asyncDelegate = delegate;
的详细实现如下
- (void)setAsyncDelegate:(id<ASTableDelegate>)asyncDelegate
{
ASDisplayNodeAssertMainThread();
NS_VALID_UNTIL_END_OF_SCOPE id oldDelegate = super.delegate;
//如果是置空,则表明是要释放delegate
if (asyncDelegate == nil) {
_asyncDelegate = nil;
_proxyDelegate = _isDeallocating ? nil : [[ASTableViewProxy alloc] initWithTarget:nil interceptor:self];
memset(&_asyncDelegateFlags, 0, sizeof(_asyncDelegateFlags));
} else {
//这里开始设置Delegate
_asyncDelegate = asyncDelegate;
//这里插入了代理
_proxyDelegate = [[ASTableViewProxy alloc] initWithTarget:_asyncDelegate interceptor:self];
//这里用于判断Delegate是否实现了如下方法。
_asyncDelegateFlags.scrollViewDidScroll = [_asyncDelegate respondsToSelector:@selector(scrollViewDidScroll:)];
_asyncDelegateFlags.tableViewWillDisplayNodeForRow = [_asyncDelegate respondsToSelector:@selector(tableView:willDisplayNode:forRowAtIndexPath:)];
_asyncDelegateFlags.tableNodeWillDisplayNodeForRow = [_asyncDelegate respondsToSelector:@selector(tableNode:willDisplayRowWithNode:)];
//这里省略一大段类似的代码
}
super.delegate = (id<UITableViewDelegate>)_proxyDelegate;
}
其中
_proxyDelegate = [[ASTableViewProxy alloc] initWithTarget:_asyncDelegate interceptor:self];
很关键,我们要注意ASTableViewProxy的实现
- (BOOL)interceptsSelector:(SEL)selector
{
return (
// handled by ASTableView node<->cell machinery
selector == @selector(tableView:cellForRowAtIndexPath:) ||
selector == @selector(tableView:heightForRowAtIndexPath:)
//这里省略一大段类似的代码
);
}
@end
interceptsSelector指的就是需要拦截的方法,并且拦截的方法要被ASTableview使用。因此,我们能在ASTableview中发现如下代码:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
而他们的实现如下:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
//设置cell
_ASTableViewCell *cell = [self dequeueReusableCellWithIdentifier:kCellReuseIdentifier forIndexPath:indexPath];
cell.delegate = self;
//创建自己的Node
ASCellNode *node = [_dataController.visibleMap elementForItemAtIndexPath:indexPath].node;
if (node) {
[_rangeController configureContentView:cell.contentView forCellNode:node];
cell.node = node;
}
return cell;
}
看到这里,想必大家知道为什么在ViewController
中能见到
- (ASCellNodeBlock)tableNode:(ASTableNode *)tableNode nodeBlockForRowAtIndexPath:(NSIndexPath *)indexPath
等代码的原因了