解決tableviewCell覆蓋的問題以及tableviewCell複用原理
先說一下解決辦法:
1.在自定義的tableviewCell的m檔案中,重寫
-(instancetype)initWithStyle:(UITableViewCellStyle)style
reuseIdentifier:(NSString *)reuseIdentifier方法,將控制元件的初
始化放在這裡方法裡面. 並自定義一個函式,為相應的控制元件賦值.
@property (strong, nonatomic) UIButton *iconBtn;
@property (strong, nonatomic) UIButton *productBtn;
@property (strong, nonatomic) UILabel *industryLbl;
@property (strong, nonatomic) UILabel *reportTimeLbl;
@property (strong, nonatomic) UILabel *companyLbl;
@property (strong, nonatomic) UILabel *titleLbl;
@property (strong, nonatomic) UILabel *srcLbl;
/**
* 重寫initWithStyle:reuseIdentifier:方法
*
* @param style
* @param reuseIdentifier
*
* @return
*/
- (instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier{
self = [super initWithStyle:UITableViewCellStyleDefault reuseIdentifier:reuseIdentifier];
if (self) {
_iconBtn = [[UIButton alloc] initWithFrame:CGRectMake(8, 8, 60, 60)];
_productBtn = [[UIButton alloc] init];
_companyLbl = [[UILabel alloc] init];
_reportTimeLbl = [[UILabel alloc] initWithFrame:CGRectMake(kScreenWidth - 78, 10, 70, 20)];
_industryLbl = [[UILabel alloc] init];
_titleLbl = [[UILabel alloc] init];
_srcLbl = [[UILabel alloc] init];
[self.contentView addSubview:_iconBtn];
[self.contentView addSubview:_productBtn];
[self.contentView addSubview:_companyLbl];
[self.contentView addSubview:_reportTimeLbl];
[self.contentView addSubview:_industryLbl];
[self.contentView addSubview:_titleLbl];
[self.contentView addSubview:_srcLbl];
}
return self;
}
/**
* 自定義函式,為cell裡面控制元件賦值
*
* @param product
*/
- (void)initData:(ProductModel *)product{
NSURL *iconUrl = [NSURL URLWithString:product.icon];
[_iconBtn setImageForState:UIControlStateNormal withURL:iconUrl placeholderImage:[UIImage imageNamed:@"loudou"]];
[_iconBtn.layer setCornerRadius:30.f];
[_iconBtn.layer setMasksToBounds:YES];
_iconBtn.layer.borderWidth = 1.f;
_iconBtn.layer.borderColor = [UIColor lightGrayColor].CGColor;
NSString *productStr = product.product;
CGFloat productX = 8 + 60 + 8;
CGFloat productW = [self calculateSize:productStr withFontSize:12.f].width;
_productBtn.frame = CGRectMake(productX, 10, productW, 20);
[_productBtn setTitle:productStr forState:UIControlStateNormal];
_productBtn.contentHorizontalAlignment = UIControlContentHorizontalAlignmentLeft;
[_productBtn setTitleColor:[[UIColor alloc] initWithRed:48/255.f green:131/255.f blue:251/255.f alpha:1.0] forState:UIControlStateNormal];
_productBtn.titleLabel.font = [UIFont systemFontOfSize:12.f];
NSString *industryStr = product.industry;
CGFloat industryX = productX +productW + 3;
CGFloat industryW = kScreenWidth - industryX - 78;
_industryLbl.frame = CGRectMake(industryX, 10, industryW, 20);
_industryLbl.text = industryStr;
_industryLbl.font = [UIFont systemFontOfSize:13.f];
_industryLbl.textAlignment = NSTextAlignmentLeft;
_reportTimeLbl.text = product.report_time;
_reportTimeLbl.font = [UIFont systemFontOfSize:12.f];
NSString *titleStr = product.title;
CGSize titleSize = [self calculateSize:titleStr withFontSize:15.f];
CGFloat titleH = titleSize.height;
CGFloat titleW = kScreenWidth - 76 - 8;
_titleLbl.frame = CGRectMake(76, 48, titleW, titleH);
_titleLbl.textColor = [[UIColor alloc] initWithRed:48/255.f green:131/255.f blue:251/255.f alpha:1.0];
_titleLbl.text = product.title;
_titleLbl.backgroundColor = [UIColor clearColor];
_titleLbl.font = [UIFont systemFontOfSize:15.f];
_titleLbl.numberOfLines = 0;
_titleLbl.lineBreakMode = NSLineBreakByWordWrapping;
_srcLbl.frame = CGRectMake(76, 48 + titleH, titleW, 20);
_srcLbl.backgroundColor = [UIColor clearColor];
_srcLbl.textColor = [UIColor grayColor];
_srcLbl.text = product.src;
_srcLbl.textAlignment = NSTextAlignmentRight;
_srcLbl.font = [UIFont systemFontOfSize:14.f];
_companyLbl.frame = CGRectMake(76, _srcLbl.frame.origin.y + 20, kScreenWidth - 76 - 8, 20);
_companyLbl.text = product.company;
_companyLbl.textAlignment = NSTextAlignmentLeft;
_companyLbl.font = [UIFont systemFontOfSize:13.f];
}
2.在tableview裡的呼叫
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
ProductModel *product = [_page.productArr objectAtIndex:indexPath.row];
_cellIdentifier = @"ReportTableViewCell";
ReportingTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:_cellIdentifier];
if (!cell) {
cell = [[ReportingTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:_cellIdentifier];
}
[cell initData:product];
return cell;
}
接下來展示一下會出現問題的程式碼:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
ProductModel *product = [_page.productArr objectAtIndex:indexPath.row];
_cellIdentifier = @"ReportTableViewCell";
ReportingTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:_cellIdentifier];
if (!cell) {
cell = [[ReportingTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:_cellIdentifier];
}
UILabel *reportTimeLbl = [[UILabel alloc] initWithFrame:CGRectMake(kScreenWidth - 78, 10, 70, 20)];
reportTimeLbl.text = product.report_time;
[cell.contentView addSubview:reportTimeLbl];
return cell;
}
因為複用,reportTimeLbl會一遍又一遍的去初始化然後新增到cell的contentView上,導致重疊.具體tableviewCell複用原理如下:
===============================這裡是分隔線,下面為轉載內容===============================
============================================================================
TableView的重用機制,為了做到顯示和資料分離,tableView
的實現並且不是為每個資料項建立一個tableViewCell。而是隻建立螢幕可顯示最大個數的cell,然後重複使用這些cell,對cell做單獨的顯示配置,來達到既不影響顯示效果,又能充分節約內容的目的。下面簡要分析一下它的實現原理。
重用實現分析
檢視UITableView標頭檔案,會找到NSMutableArray* visiableCells,和NSMutableDictnery* reusableTableCells兩個結構。visiableCells內儲存當前顯示的cells,reusableTableCells儲存可重用的cells。
TableView顯示之初,reusableTableCells為空,那麼tableView dequeueReusableCellWithIdentifier:CellIdentifier返回nil。開始的cell都是通過[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier]來建立,而且cellForRowAtIndexPath只是呼叫最大顯示cell數的次數。
比如:有100條資料,iPhone一屏最多顯示10個cell。程式最開始顯示TableView的情況是:
1. 用[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier]建立10次cell,並給cell指定同樣的重用標識(當然,可以為不同顯示型別的cell指定不同的標識)。並且10個cell全部都加入到visiableCells陣列,reusableTableCells為空。
2. 向下拖動tableView,當cell1完全移出螢幕,並且cell11(它也是alloc出來的,原因同上)完全顯示出來的時候。cell11加入到visiableCells,cell1移出visiableCells,cell1加入到reusableTableCells。
3. 接著向下拖動tableView,因為reusableTableCells中已經有值,所以,當需要顯示新的cell,cellForRowAtIndexPath再次被呼叫的時候,tableView dequeueReusableCellWithIdentifier:CellIdentifier,返回cell1。cell1加入到visiableCells,cell1移出reusableTableCells;cell2移出visiableCells,cell2加入到reusableTableCells。之後再需要顯示的Cell就可以正常重用了。
所以整個過程並不難理解,但需要注意正是因為這樣的原因:配置Cell的時候一定要注意,對取出的重用的cell做重新賦值,不要遺留老資料。
一些情況
使用過程中,我注意到,並不是只有拖動超出螢幕的時候才會更新reusableTableCells表,還有:
1. reloadData,這種情況比較特殊。一般是部分資料發生變化,需要重新重新整理cell顯示的內容時呼叫。在cellForRowAtIndexPath呼叫中,所有cell都是重用的。我估計reloadData呼叫後,把visiableCells中所有cell移入reusableTableCells,visiableCells清空。cellForRowAtIndexPath呼叫後,再把reuse的cell從reusableTableCells取出來,放入到visiableCells。
2. reloadRowsAtIndex,重新整理指定的IndexPath。如果呼叫時reusableTableCells為空,那麼cellForRowAtIndexPath呼叫後,是新建立cell,新的cell加入到visiableCells。老的cell移出visiableCells,加入到reusableTableCells。於是,之後的重新整理就有cell做reuse了。