TableView的一些坑

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相关内容会持续更新,欢迎大家批评指正。

你可能感兴趣的:(TableView的一些坑)