- 系统如何计算的自适应高度?
- 系统计算的行高会不会被缓存?
- 如何缓存?
- 勘误
前几天读文档的时候发现一对方法
- (CGSize)systemLayoutSizeFittingSize:(CGSize)targetSize NS_AVAILABLE_IOS(6_0);
- (CGSize)systemLayoutSizeFittingSize:(CGSize)targetSize withHorizontalFittingPriority:(UILayoutPriority)horizontalFittingPriority verticalFittingPriority:(UILayoutPriority)verticalFittingPriority NS_AVAILABLE_IOS(8_0);
具体可以参阅《iOS文档补完计划--UIView》中的相关解释。
简而言之这两个方法会:
返回Auto Layout后内容高度
并且、我们都知道UITableView
、如果设置成rowHeight = UITableViewAutomaticDimension
的话。cell的高度将由系统通过Auto Layout
自动计算。
-
系统如何计算的自适应高度?
而这个计算、是否通过上面两个方法呢?
经过试验、答案是肯定的。
系统调用的正是- (CGSize)systemLayoutSizeFittingSize:(CGSize)targetSize withHorizontalFittingPriority:(UILayoutPriority)horizontalFittingPriority verticalFittingPriority:(UILayoutPriority)verticalFittingPriority NS_AVAILABLE_IOS(8_0);
这个方法。
-
系统计算的行高会不会被缓存?
经过试验、答案是否定的。也就是系统不会缓存计算过的行高
这里有两个能够让Cell自适应的方式
- 对UITableView进行设置
tableView.rowHeight = UITableViewAutomaticDimension
- 通过代理返回
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
return UITableViewAutomaticDimension;
}
结果是无论使用哪个方法、在每次Cell即将被展示的时候、都会自动调用上述的systemLayoutSizeFittingSize
方法。
两个关键的步骤是:
- 通过
cellForRowAtIndexPath
对某个Cell进行配置
而我们在这一步已经将Cell的内容配置完毕了 - 通过
[UITableView _heightForCell:atIndexPath:]
计算Cell高度
而内部则调用systemLayoutSizeFittingSize
获取具体的高度。
-
如何缓存?
经过以上两个探索、我们已经知道Cell通过systemLayoutSizeFittingSize
高度、并且不会被缓存。
那么、我们需要做的就是自己计算高度、并且缓存。直接贴一下代码:
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
BSQuestionsModel * model = _dataArray[indexPath.section];
return model.cell_height?:UITableViewAutomaticDimension;
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
BSQuestionsModel * model = _dataArray[indexPath.section];
BSQuestionsTableViewCell * cell = [BSQuestionsTableViewCell cellForTableView:tableView model:model];
//高度缓存
CGFloat height = [cell systemLayoutSizeFittingSize:CGSizeMake(tableView.frame.size.width, 0) withHorizontalFittingPriority:UILayoutPriorityRequired verticalFittingPriority:UILayoutPriorityFittingSizeLevel].height;
model.cell_height = height;
return cell;
}
这样、cell在进行过一次高度计算之后。就不需要在计算第二次了
然后关于上面的代码有几点需要说:
- 为什么在
cellForRowAtIndexPath
里做缓存
最开始我们已经谈过了、cellForRowAtIndexPath
的调用在获取自动布局的高度之前、这样也能避免重复取用对应位置的Cell。前提是你开启了预估行高、具体见下面的解释。
这里需要补充一下
cellForRowAtIndexPath
与heightForRowAtIndexPath
调用到底谁先谁后:
- 如果你的tableView设置了预估行高
cellForRowAtIndexPath
永远在heightForRowAtIndexPath
之前- 如果你的tableView没有设置预估行高
tableView首先会把所有IndexPath的heightForRowAtIndexPath
轮训一遍以计算contentSize
。而后按照情况1的顺序。
所以如果你想不设置预估行高、又想用这种方式缓存的话。需要把缓存的代码写在heightForRowAtIndexPath
而不是cellForRowAtIndexPath
里
而返回的UITableViewAutomaticDimension
主要是为了容错(比如上面的情况)。
- 为什么用
systemLayoutSizeFittingSize:withHorizontalFittingPriority:verticalFittingPriority
网上很多帖子都这样写:
[cell.contentView systemLayoutSizeFittingSize:UILayoutFittingCompressedSize]
但是在我这不太好用、因为我cell内部有一些优先级的设置。
所以、我干脆和系统调用的方式一样。
- 异步计算
是的、我们又可以异步计算了。虽然我没写、因为我现在得抓紧码页面~
关于一些旧帖子
我搜到的都是14/15年的帖子、和现在的情况感觉还是有出入的。
-
cell.contentView
取出的高度要+1
网上对+1的解释是、cell
比cell.contentView
要搞出1个单位。还附上了两张图。
但是现在、cell是比cell.contentView高出0.5(0.5也不一定准确、xib上有四舍五入的嫌疑)、而不是1。
- 用
cell
还是用cell.contentView
我在网上搜了很多帖子、都说要使用cell.contentView
但是我用cell一样可以获取高度。所以用cell呗~
勘误
不好意思今天是2019.11.29,时隔一年半了才发现这个事。因为看了Bruce_XHG大佬的文章
不过其实也不算是勘误、只是发现个更好的方法
由于我以前很少使用willDisplayCell
这个方法(其实现在也是),所以一直没发现:
实际上对于自适应的cell,只要你在cellForRowAtIndexPath
里配置好了,willDisplayCell
中返回的cell其实已经被系统计算过了。
我们可以省去自己计算的部分,也不需要担心自己算的不对。上面的反面教材就不删了
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
GKTMessageItemModel *model = self.dataArr[indexPath.row];
UITableViewCell<GKTMessageListTableViewCellProtocol> *cell = [self factoryCellWithTableView:tableView model:model row:indexPath.row];
//这个页面对cell与model进行了进一步的解耦,改的时候如果对逻辑不清晰,可以参阅。
////www.greatytc.com/p/723e8435586d#comment-38321201
[cell configCellWithModel:model indexPath:indexPath];
return cell;
}
- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath {
GKTMessageItemModel *model = self.dataArr[indexPath.row];
//高度缓存
if (model.cell_height == 0) {
CGFloat height = cell.height;
model.cell_height = height;
}
}
最后
本文主要是自己的学习与总结。如果文内存在纰漏、万望留言斧正。如果愿意补充以及不吝赐教小弟会更加感激。