关于UITableView的使用方法有很多,本文主要通过代码说明UITableViewCell的加载方式和重用机制。
关于重用机制,网上有各种解释,只要记住用tableview就得用重用机制就可以了。
下面将介绍几种常见的cell加载方式。
自定义tableview的cell为纯代码方式
<a name="cell-code-one"></a>方法一:
#define IDENTIFIER @"identifier"
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
UITableViewCell* cell = nil;
//使用重用机制,IDENTIFIER作为重用机制查找的标识,tableview查找可用cell时通过IDENTIFIER检索,如果有则cell不为nil
cell = [tableView dequeueReusableCellWithIdentifier:IDENTIFIER];
if (cell == nil) {
cell = [[CustomTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:IDENTIFIER];//代码创建的cell需要使用该初始化方法
}
return cell;
}
以上方式要注意cell创建时的Identifier与tableview dequeueReusableCell时是一样的。
创建完成后,运行效果如下:
QQ20160328-0.png
cell共被创建了15次,但后期再怎么滑也不会创建新的cell了
<a name="cell-code-two"></a>方法二:
#define IDENTIFIER @"identifier"
首先在viewdidload中注册cell
- (void)viewDidLoad {
[super viewDidLoad];
[self.tableView registerClass:[CustomTableViewCell class] forCellReuseIdentifier:IDENTIFIER];
}
然后在tableview的delegate中
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
UITableViewCell* cell = nil;
cell = [tableView dequeueReusableCellWithIdentifier:IDENTIFIER];
//这里会发现,cell一直都不是nil,不再需要创建
if (cell == nil) {
NSLog(@"cell create at row:%ld", (long)indexPath.row);
}
return cell;
}
通过注册cell类的方式,让tableview自己加载,不再需要手动创建cell,这里与UICollectionViewCell类似,有兴趣的可以了解一下,运行后效果如下:
QQ20160328-1.png
可以看出,运行后,tableView dequeueReusableCellWithIdentifier总是有值,不再需要cell创建,似乎更简洁方便了。
自定义tableview的cell为xib方式
方法一:
#define IDENTIFIER @"identifier"
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
UITableViewCell* cell = nil;
cell = [tableView dequeueReusableCellWithIdentifier:IDENTIFIER];
if (cell == nil) {
cell = [[NSBundle mainBundle] loadNibNamed:@"CustomTableViewCell" owner:nil options:nil].firstObject;
NSLog(@"cell create at row:%ld", (long)indexPath.row);//此处要使用loadnib方式!
}
return cell;
}
CustomTableviewCell为xib的名字,xib中的identifier必须与代码中的IDENTIFIER定义相同,运行效果如下:
QQ20160328-2.png
可以看到,cell是从xib加载的,cell创建现象与代码创建相同。
方式二:
#define IDENTIFIER @"identifier"
首先在viewdidload中注册cell
- (void)viewDidLoad {
[super viewDidLoad];
[self.tableView registerNib:[UINib nibWithNibName:@"CustomTableViewCell" bundle:nil] forCellReuseIdentifier:IDENTIFIER];
}
然后在tableview的delegate中
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
UITableViewCell* cell = nil;
cell = [tableView dequeueReusableCellWithIdentifier:IDENTIFIER];
if (cell == nil) {
//这里会发现,cell一直都不是nil,不再需要创建
NSLog(@"cell create at row:%ld", (long)indexPath.row);
}
return cell;
}
通过注册cell类的方式,让tableview自己加载,不再需要手动创建cell,运行效果如下:
QQ20160328-3.png
可以看到,cell是从xib加载的,cell创建现象与cell类注册相同。
通过上述重用方式看,通过tableView的registerNib货registerClass的方式显得更加简单鲜明,且不易出错。至于这玩意什么时候出现的我也不清楚了。。。
重用机制的必要性
有的同学说,我的tableview固定只有20个cell,不会一直加载,为什么要使用重用机制?其实这个同学就是我。。。那么我们来一下不使用重用机制的情况下对cell的创建情况。
#define ROWS_COUNT 20 //定义tableview只有20条数据
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
UITableViewCell* cell = nil;
cell = [tableView dequeueReusableCellWithIdentifier:@""];
if (cell == nil) {
cell = [[CustomTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:IDENTIFIER];//代码创建的cell需要使用该初始化方法
}
return cell;
}
运行后,我上下滑动tableview,效果如下:
QQ20160328-4.png
可以看到,只要当有新的cell显示时,cell就会重新创建,所以虽然共有20条数据,但是来来回回不停滑动会让cell一直重复创建,创建多了就会有问题
一个xib文件,多个cell的情况
有的同学想把多个自定义cell放到一个xib中,以减少xib文件的个数,这个同学还是我。。。对于这种情况,我只找到了loadNib方式加载cell的方法,但是对于registerNib的方法好像不适用,请指教。
在一个xib文件中创建两个cell:
QQ20160328-5.png
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
UITableViewCell* cell = nil;
cell = [tableView dequeueReusableCellWithIdentifier:IDENTIFIER];
if (cell == nil) {
//通过loadNib找到的对象是个array,当前情况为两个cell
NSArray* objects = [[NSBundle mainBundle] loadNibNamed:@"CustomTableViewCell" owner:nil options:nil];
//遍历找到CustomTableViewCell
for (id object in objects) {
if ([object isKindOfClass:[CustomTableViewCell class]]) {
cell = object;
break;
}
}
NSLog(@"cell create at row:%ld", (long)indexPath.row);
}
NSNumber* number = self.array[indexPath.row];
cell.textLabel.text = number.stringValue;
return cell;
}
可以看到,TableViewCell被成功加载,运行效果如下:
QQ20160328-6.png
以上有不准确的地方请大伙儿指正!
网友评论
已修改,UITableViewCell->TableViewCell,其实主要是想说UITableviewCell的加载和重用机制,可能名字起的不算好~ 感谢指正
另外数据源方法也是代理的方式,这里只能算是说法不同吧