背景
嗯。。。因为懒,总是写 UITableView 的代理方法真的很烦,尤其是表格界面。
先抛出代码。
一、使用超简单
想要的就是这个样子的:
pod `LCEasyTableView`
使用时:
self.tableView.lc_dataArray = dataArray;
二、主要技术
2.1 提出问题
UITableView
常用的几个代理方法真的是写烦了,怎样摆脱这几个代理方法呢?
有的继承UITableView
,有的在UITableView
外包装一层,但是偶尔UIViewController
外部需要重写这几个代理方法,还要把代理方法抛出去,这样就实现很多个UITableView
的代理方法,然后使用自己的代理抛出去,实在繁琐,也不易扩展。
例如下面这个样子:
都说Categroy
好,要怎样利用它来更好的实现代理方法的分配呢?
2.2 原理
大家都知道报unrecognized selector
的异常之前,有四次机会拯救崩溃:
resolveInstanceMethod:
或resolveClassMethod:
允许用户在此时为该 Class 动态添加实现;forwardingTargetForSelector:
尝试找到一个能响应该消息的对象;methodSignatureForSelector:
尝试获得一个方法签名;forwardInvocation:
处理第3步获取到的方法签名包装成的Invocation
。
以上四步仍然失败,调用doesNotRecognizeSelector: 抛出异常。
其原理就是objc_msgForward
消息转发过程。
2.3 方案
摆脱代理方法的
TableView
封装:
- 在不使用继承,不影响原本功能的影响之外,使用
Category
的方式处理数据赋值; - 使用
NSProxy
进行代理方法的分发,既摆脱繁琐的写代理方法,又在需要的地方不影响使用。
各类之间关系图如下:
在这里,也是使用了消息转发原理:
- LCTableViewIMP:实现最常用的几个代理方法;
- UIViewController/UIView:表示想额外实现代理方法的类;
- UITableView+LCAdd:使用扩展灵活性高;
- LCTableViewProxy 的实现:
- 接收
UITableView+LCAdd
的代理方法; - 弱引用
LCTableViewIMP
和UIViewController/UIView
(或其他额外想实现代理方法的类)两个对象; - 实现
respondsToSelector:
判断LCTableViewIMP
和UIViewController
对象是否实现了该方法; - 实现
methodSignatureForSelector:
获得一个方法签名; - 实现
forwardInvocation:
处理把代理方法分配给LCTableViewIMP
或UIViewController
,以UIViewController
优先。
这样,不使用继承、不封装一系列代理方法,仅在UITableView
原来的使用上额外的LCTableViewIMP
实现常用代理方法,使UIViewController
在不实现那几个常用的代理方法下,也可以正常显示,即摆脱TableView的代理方法。
2.4 使用
使用很简单,Category 里主要提供了 lc_dataArray 用于数据赋值、lc_Delegate 额外使用代理。
@property (nonatomic, weak) id<UITableViewDelegate, UITableViewDataSource> lc_Delegate;
@property (nonatomic, strong, readonly) NSMutableArray *lc_dataArray;
2.5 数据需要遵循协议
有方便必然有规则,代理方法的默认实现必要得到适当的数据,所以这里需要数据遵循一定的协议LCCellDataProtocol
:
@optional
@property (nonatomic, assign) CGFloat cellHeight;
/** 可用于数组多种cell样式,增加类型方便处理逻辑 */
@property (nonatomic, assign) NSInteger identityType;
/** 需提供cell类 */
+ (Class)cellClass;
LCCellDataProtocol
协议规定 cell 的数据(ViewModel/Model
)提供该信息。
三、实现功能
3.1 常用数据格式的展示
- 一个section,多个row;
- 多个section,一个row;
- 多个section,多个row。
其中,1和2的区分使用 Category 的 lc_isSectionsStyle 属性区分。
3.2 支持不同cell
写法
- 使用 model 赋值;
- MVVM 使用 viewModel 赋值;
- 使用 Xib 的 cell。
且支持多种 cell 的一个数组数据。
3.3 不同cell
样式
每个项目都有常用的 3 种 cell 样式:展示/点击、左提示右内容、左提示右输入。考虑不同项目样式区别太大,即便是腾讯的 QMUI 封装的用起来也不方便。所以没有放到 pod 中,作为参考放在 Example 里哈。
这是额外自己封装的一个常用的点击样式 cell 。