UITableView使用AutoLayout动态计算cell高度

UITableView几乎是每个app都需要用的控件,而cell高度自适应也是我们 需要掌握的,当然cell上面的控件也是有多种表现形式,今天小编讲解的是其中一种比较常见的:Auto Layout with UILabel in UITableViewCell。
话不多说,上教程。
首先我们创建一个Sigle View Application的项目,然后拖上一个UITableView到storyboard中,设置好代理并且设置好约束。约束这里就不做讲解了,可以到AutoLayout详解了 解一下,接着回到我们项目,我们创建一个继承UITableViewCell的类,此处我命名为DIYTableViewCell,接着在 storyboard中拖一个UITableViewCell到UITableView上,然后把这个cell关联我们创建的 DIYTableViewCell类,什么?不会关联?没关系,看这里!
这里写图片描述

然后给cell一个复用的标识,这里我命名为cell
这里写图片描述

接下来我们在cell上面放置一个label和一个UIImageView并且设置好约束,此处注意,应该把UILabel的numberOfLines属性设置为0.
现在我们应该给ViewController加点料了,回到我们的ViewController,实现UITableView的代理方法,

`#pragma mark - UITableViewDataSource
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    return self.tbData.count; } - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { DIYTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:iden]; cell.titleLabel.text = [self.tbData objectAtIndex:indexPath.row]; return cell; }`
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

这里有一个需要特别注意的问题,也是效率问题。UITableView是一次性计算完所有Cell的高度,如果有100个Cell,那么- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath就会触发100次,然后才显示内容。不过在iOS7以后,提供了一个新方法可以避免这100次调用,它就是- (CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath。 要求返回一个Cell的估计值,实现了这个方法,那只有显示的Cell才会触发计算高度的protocol. 由于systemLayoutSizeFittingSize需要cell的一个实例才能计算,所以这儿用一个成员变量存一个Cell的实列,这样就不需 要每次计算Cell高度的时候去动态生成一个Cell实例,这样即方便也高效也少用内存,可谓一举三得。关于UITableViewCell的优化,可以 看这里优化UITableViewCell高度计算

接下来我们声明一个这样的实例变量:

@property (nonatomic, strong) DIYTableViewCell *cell;
  • 1

然后实例化它:

self.cell = [self.tableView dequeueReusableCellWithIdentifier:iden];
  • 1

下面是计算cell高度的代码:

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{

    DIYTableViewCell *cell = (DIYTableViewCell *)self.cell; cell.titleLabel.text = [self.tbData objectAtIndex:indexPath.row]; CGSize size = [cell.contentView systemLayoutSizeFittingSize:UILayoutFittingCompressedSize]; NSLog(@"调用次数%ld",indexPath.row+1); return MAX(44, 1 + size.height) ; }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

可能你会觉得奇怪,为什么需要加1呢?笔者告诉你,如果不加1,结果就是错误的,Cell中UILabel将显示不正确。原因就是因为这行代码 CGSize size = [cell.contentView systemLayoutSizeFittingSize:UILayoutFittingCompressedSize];由于是在 cell.contentView上调用这个方法,那么返回的值将是contentView的高度,UITableViewCell的高度要比它的 contentView要高1,也就是它的分隔线的高度。
到这里我们的cell自适应高度就算完成了,看看我们的运行结果吧:
这里写图片描述
Domo源码已经上传,如有需要可以下载AutoLayoutCell

 

转载于:https://www.cnblogs.com/duyuiOS/p/4968136.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值