【iOS】自定义cell及其复用机制
文章目录
- cell的复用
- 注册
- 非注册
- 两者的区别
- 自定义cell
cell的复用
当用户滚动 UITableView 或 UICollectionView 时,只有少量可见的 cell 会被实际创建和显示。对于那些暂时不可见的 cell,系统会将它们缓存起来以备将来复用。这就是所谓的 cell 复用机制。
为什么需要cell的复用:
- 提高性能: 不需要为每个 cell 都创建一个新的实例,减少了创建 cell 的开销,从而提高了滚动的流畅度。
- 节省内存: 只维护少量可见 cell 的实例,而不是为整个数据集创建 cell,从而大大节省了内存使用。
- 简化代码: 通过复用 cell,我们只需要更新 cell 的内容,而不需要频繁地创建和销毁 cell 实例。
实现cell的复用有两种方法,一种是手动判空实现,一种是使用cell的注册机制
注册
- 注册单元格是在viewDidLoad中使用register(_:forCellReuseIdentifier:)方法来完成的
- 在创建cell的函数中使用dequeueReusableCellWithIdentifier获取可复用的cell。(如果没有可复用的cell,就自动利用注册cell时提供的类创建一个新的cell并返回)
示例程序:
- (void)viewDidLoad { [super viewDidLoad]; // 使用代码自定义 Cell [self.tableView registerClass:[CustomCell class] forCellReuseIdentifier:@"id"]; } - (UITableViewCell*)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { static NSString *strID = @"id"; MyTableViewCell *cell = [self.tableView dequeueReusableCellWithIdentifier: strID]; return cell; }
非注册
- 非注册方法是直接通过实例化单元格对象,并在需要时手动创建和配置每个单元格
- 每次需要显示新的单元格时,都会实时创建新的单元格对象,不会尝试重用已存在的单元格
示例程序:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { static NSString *identifier = @"mycell"; UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier]; if (!cell) { cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:identifier]; } return cell; }
两者的区别
- 注册方法的代码更简洁,不需要手动检查 cell 是否为 nil,非注册方法需要更多的判断和创建新 cell 的代码。
- 注册方法在获取可复用 cell 时更高效,因为系统可以直接从复用池中获取,非注册方法需要手动检查 cell 是否为 nil,并在需要时创建新的 cell,会稍微慢一些。
自定义cell
自定义 cell 可以让你更好地控制 cell 的外观和行为,提高代码的可读性和可维护性。同时,合理的复用机制也可以显著提高滚动性能。
自定义cell的具体步骤:
- 创建自定义cell类
- 添加UI元素
- 实现初始化方法
- 设置cell的布局
- 在TableView中使用自定义cell
示例程序:
**先自己创建一个MyCell类,继承于UITableViewCell类,在该类中定义自己将使用的控件,并规定他们的位置等信息
MyCell.h
#import NS_ASSUME_NONNULL_BEGIN @interface MyCell : UITableViewCell @property UIButton* btn; @property UILabel* label; @property (nonatomic, strong) UISwitch* swt; @end
MyCell.m
#import "MyCell.h" @implementation MyCell -(instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier { self = [super initWithStyle:style reuseIdentifier:reuseIdentifier]; if([self.reuseIdentifier isEqualToString:@"mycell"]) { _label = [[UILabel alloc] init]; [self.contentView addSubview:_label]; _swt = [[UISwitch alloc] init]; _btn = [[UIButton alloc] init]; [self.contentView addSubview:_btn]; } return self; } -(void)layoutSubviews{ _btn.frame = CGRectMake(0, 0, 50, 50); _label.frame = CGRectMake(60, 0, 100, 50); } @end
ViewContreller.m:
#import "ViewController.h" #import "MyCell.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view. self.view.backgroundColor = [UIColor whiteColor]; self.tableView = [[UITableView alloc] initWithFrame:self.view.bounds style:UITableViewStylePlain]; [self.view addSubview: self.tableView]; self.tableView.dataSource = self; self.tableView.delegate = self; [self.tableView registerClass:[MyCell class] forCellReuseIdentifier:@"mycell"]; } - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView { return 2; } - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { return 4; } - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath { return 120; } - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { MyCell* cell = [self.tableView dequeueReusableCellWithIdentifier:@"mycell" forIndexPath:indexPath]; if(indexPath.section == 1) { cell.label.text = @"Lee"; [cell.btn setImage:[UIImage imageNamed:@"12.png"] forState:UIControlStateNormal]; } else { cell.label.text = @"Xxx"; [cell.btn setImage:[UIImage imageNamed:@"im1.jpg"] forState:UIControlStateNormal]; } return cell; } @end
运行结果
文章版权声明:除非注明,否则均为主机测评原创文章,转载或复制请以超链接形式并注明出处。