tableView是iOS页面中使用最频繁的类,没有之一。在我目前的项目中,95%的页面都是以tableView为基础开发的。在开发中学习,在学习中开发,下面分享一些坑,和自己的领悟。欢迎讨论指正 ... ...
tableView中cell的复用问题
- 1.一般情况下是这样的,几乎成了套路。(个人建议在开发中将代理部分写成代码块,几个字母就能完成,能够大大提高开发效率)
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
MyOrderCell *cell = [tableView dequeueReusableCellWithIdentifier:@"MyOrderCell"];
if (!cell) {
cell = [[MyOrderCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"MyOrderCell"];
}
cell.selectionStyle = UITableViewCellSelectionStyleNone;
cell.orderModel = self.dataArray[indexPath.section];
return cell;
}
两个注册方法
// Used by the delegate to acquire an already allocated cell, in lieu of allocating a new one.
获取一个已经分配的cell,替代创建一个新的
- (nullable __kindof UITableViewCell *)dequeueReusableCellWithIdentifier:(NSString *)identifier;
// newer dequeue method guarantees a cell is returned and resized properly, assuming identifier is registered
新的出队方法保证一个cell被返回并适当调整,如果关键字被注册
- (__kindof UITableViewCell *)dequeueReusableCellWithIdentifier:(NSString *)identifier forIndexPath:(NSIndexPath *)indexPath NS_AVAILABLE_IOS(6_0);
第二个方法是在iOS 6.0之后出现的,但是如果使用该方法,就必须保证identifier在storyboard中存在,否则会崩溃。报错如下:reason: 'unable to dequeue a cell with identifier AgentEmployeeTracking - must register a nib or a class for the identifier or connect a prototype cell in a storyboard'
初始化方法
// Designated initializer. If the cell can be reused, you must pass in a reuse identifier. You should use the same reuse identifier for all cells of the same form.
指定的初始化方法。如果这个cell可以复用,你必须传入一个复用标识,你必须使用相同的复用标识为所有相同(样式)的行。
- (instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(nullable NSString *)reuseIdentifier NS_AVAILABLE_IOS(3_0) NS_DESIGNATED_INITIALIZER;
保证初始化方法标识reuseIdentifier和上面的注册方法一致即可。其次,开发中注意,如果系统样式可以满足需求,尽量用系统的默认样式,提高运行效率。
model的传递
将model设置为新建cell的属性,重写model的set方法,在set方法中将值写入到cell的相关控件中;其次,对cell一些属性样式的设置,也建议放到cell中,让MVC各司其职,尽量减少在其他页面暴露不相干的内容。
- (void)setOrderModel:(AgentMyOrderModel *)orderModel{
[self.faileInBtn setTitle:[NSString stringWithFormat:@"淘汰\n%@人",orderModel.failNum] forState:UIControlStateNormal];
[self setButtonStyle:self.faileInBtn];
[self.faileInBtn setTag:2];
[self.faileInBtn addTarget:self action:@selector(setButtonSelectedState:) forControlEvents:UIControlEventTouchUpInside];
}
- (void)setButtonStyle:(PublicNoticeButton *)sender{
sender.borderColor = [UIColor redColor];
[sender setDisplayBoarder:YES];
[sender setTitleColor:[UIColor whiteColor] forState:UIControlStateSelected];
[sender setBackgroundImage:[UIImage imageWithColor:[UIColor redColor]] forState:UIControlStateSelected];
}
- 2.干货来了,新建cell的复用问题
在开发中很多时候我们需要自己来初始化控件,重新布局,尤其是在用代码创建控件时,可能会写成这样:
- (UserInfoEditCell *)initWithTitle:(NSString *)title IndexPath:(NSIndexPath *)indexPath inforDict:(NSDictionary *)dict{
NSString *key = [NSString stringWithFormat:@"%ld",indexPath.section*10+indexPath.row];
self.userInfoLab.text = title;
self.userInfoTextField.text = [dict objectForKey:key];
//第一行显示图像
if (indexPath.section==0) {
[self.userInfoTextField setHidden:YES];
UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(SCREEN_WIDTH-110, 5, 80, 80)];
imageView.clipsToBounds = YES;
imageView.layer.cornerRadius = 40;
[imageView sd_setImageWithURL:[NSURL URLWithString:[dict objectForKey:key]] placeholderImage:[UIImage imageNamed:@"mrtx"]];
[self.contentView addSubview: imageView];
self.accessoryType = UITableViewCellAccessoryDisclosureIndicator;
self.selectionStyle = UITableViewCellSelectionStyleNone;
}
}
上面这段代码有两个问题:1、每次调用该方法都会重新创建一个imageView对象,造成资源浪费,没有复用;2、样式相关的属性设置放置在该方法中,每次都要重复设置,不符合MVC思想。
重写初始化方法,优化如下:
- (instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier{
UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(SCREEN_WIDTH-110, 5, 80, 80)];
imageView.clipsToBounds = YES;
imageView.layer.cornerRadius = 40;
self.headerImg = imageView;
[self.contentView addSubview:self.headerImg];
return self;
}
注意:如果各行样式不同,比如只有第一行显示该imageView,则重写创建方法时添加 [self.headerImg setHidden:YES]; 在自定义方法中 [self.headerImg setHidden:NO]; 即可。
- 3.点击cell上控件时,获取当前cell的indexPath值。核心思想,还是用系统提供的方法根据点击的point获取该cell所在的indexPath。
相关方法及应用:
//将像素point从view中转换到当前视图中,返回在当前视图中的像素值
- (CGPoint)convertPoint:(CGPoint)point toView:(nullable UIView *)view;
- (CGPoint)convertPoint:(CGPoint)point fromView:(nullable UIView *)view;
//根据传入的point获取到当前的indexPath值
- (nullable NSIndexPath *)indexPathForRowAtPoint:(CGPoint)point; // returns nil if point is outside of any row in the table
- (IBAction)statusBtn:(PublicNoticeButton *)sender {
//使用这两种方法,结果都是一样的
// CGPoint clickPoint = [self.tableView convertPoint:sender.center fromView:sender.superview];
CGPoint clickPoint = [sender.superview convertPoint:sender.center toView:self.tableView];
//打印point的方法
NSLog(@"%@",NSStringFromCGPoint(clickPoint));
NSIndexPath *currentPath = [self.tableView indexPathForRowAtPoint:clickPoint];
}
注意:若该方法中传值为nil也是不会报错的,如果运行时发现异常,注意打印point值是否正确;如fromView是nil,则返回CGrectZero;如果toView是nil则相当于:[fromView convertPoint:aPoint toView:selfView.window];。
iOS相关内容会持续更新,欢迎大家批评指正。