假如我们要用tableview做一个数据表单,数据相当多。如果没有单元格的复用的话,有多少数据我们就要初始化多少个cell,这样相当消耗性能,而且意义不大--有些cell还没有显示出来。
为了解决这一问题,苹果公司已经为我们解决。
在使用tableview的时候,只有当cell将要显示出来的时候才会调用tableView:cellForRowAtIndexPath:方法,也就是说只有在屏幕上显示出来的cell才会被创建出来,没显示的cell则不会创建。
假如手机屏幕只能显示出6个cell,当向上滑动时,第一个cell将要消失,第七个cell将要显示出来的时候,创建第七个cell。当第七个cell出现时,第一个cell被销毁。以此类推,创建将要出现的新cell,销毁消失的cell。
综上所述,这样即使不需要创建多个cell,但是要经过 创建-销毁的过程,还是会消耗性能。
但苹果又为我们解决了这一问题----cell的复用
单元格复用的好处:
- 还是上面的例子,当第七个cell将要显示,第一个cell将要消失的时候,并不会销毁第一个cell,而是把第一个cell放在了缓冲区;这时第七个cell不是先创建新的cell,而是先去缓冲区中找能复用发的cell,有能复用的cell,直接拿过来用;要是没有能复用的cell,再选择创建。这样能保证创建最少(实际屏幕所能显示出的cell)的cell,来显示所有的数据。
实现cell复用代码:
- 手动创建cell
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"cell"];
if (!cell)
{
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:cellIdentifier];
cell.backgroundColor = [UIColor redColor];
}
cell.textLabel.text = [NSString stringWithFormat:@"第%ld行",indexPath.row];
return cell;
}
注意点:
当 !cell 时,创建cell时不能用 init方法,要用 initWithStyle:reuseIdentifier:方法,主要是因为我们要给cell添加上标记(reuseIdentifier),这样我们去缓冲区找能复用cell时才能找到;如果使用init方法,cell则没有标记,去缓冲区找复用cell时,则会找不到。
一般在if(!cell)中,在新创建cell的时候,将一些只需要初始化一次的属性进行初始化,而不是在这个括号的外面。因为括号外面会执行多次,而这些属性并不需要多次设置。同样,如果不同的cell需要设置不同属性或数据,那么需要在括号外执行,因为括号外面每次cell出现都会执行到,这样可以保证不用的cell对应不同的属性或数据。如果将本该设置不同cell对应不同属性的代码放在括号里面,在复用cell的时候不会覆盖这些数据,会出现不正确的结果,造成数据冗余的问题
自动创建cell
- (void)viewDidLoad
{
[super viewDidLoad];
[self.tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:@"cell"];
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"cell"];
cell.textLabel.text = [NSString stringWithFormat:@"第%ld行",indexPath.row];
return cell;
}
- 自动创建单元格,首先要注册单元格类,其实是让tableview创建cell的时,先要去缓冲区中找有没有可重用的cell,如果有,拿来直接用;如果没有,则根据注册的单元格类生成cell,并标记上identifier。 但是不能指定cell的样式,只能是默认样式。
- 这种重用方法更多用在我们自定义cell上,即使不能指定样式,但是在注册单元格类的时候,我们可以注册我们自定义cell类,这样返回时,也是我们自定义的cell。