目录
- 系统如何计算的自适应高度?
- 系统计算的行高会不会被缓存?
- 如何缓存?
前几天读文档的时候发现一对方法
- (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的顺序。具体可以参阅《iOS-谈谈UITableView中estimatedRowHeight到底该不该禁用》
所以如果你想不设置预估行高、又想用这种方式缓存的话。需要把缓存的代码写在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呗~
最后
本文主要是自己的学习与总结。如果文内存在纰漏、万望留言斧正。如果愿意补充以及不吝赐教小弟会更加感激。