今天做設備適配的時候遇到一個問題,一個有tableview的頁面上,iphone5,iphone5s,iphone6,iphone6+展現都沒有問題,但在iPhone4s上展現時發現tableview的最後一列出現內容重複的現象。數組
怎麼會出現這種問題呢:iphone
因而咱們看看代碼實現:ide
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPathspa
{orm
static NSString *cellIdentifier = @"UserInfoCell";ip
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier];rem
if (cell == nil) {string
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier];it
}io
if (indexPath.section == 0) {
if (indexPath.row == 0) {
cell.textLabel.text = @"頭像";
cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;
self.photoButton.frame = CGRectMake(self.view.frame.size.width - WIDTH_TO_LEFT_MARGIN- HEIGHT_OF_PHOTO, (HEIGHT_OF_PHOTO_CELL-HEIGHT_OF_PHOTO)/2, HEIGHT_OF_PHOTO, HEIGHT_OF_PHOTO);
UIImage *image = [UIImage imageWithContentsOfFile:self.photoGraphPath];
[self.photoButton setImage:image forState:UIControlStateNormal];
[cell.contentView addSubview:self.photoButton];
}
else if (indexPath.row == 1) {
cell.textLabel.text = @"名稱";
cell.accessoryType = UITableViewCellAccessoryNone;
[self settingLabel:self.nameLabel];
[cell.contentView addSubview:self.nameLabel];
}...............
看下代碼咱們分析一下就會知道,這是因爲cell重用引發的,在另外幾個設備上面基本上是一頁或者一頁多一點展現,所用到的cell都是直接初始化出來的,只有這個iphone4s超過一頁顯示,使用的是重用的cell,此cell上有老的數據沒有remove。
因而咱們再把cell重用的機制梳理一下:
tableview的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了。
看到這裏咱們就基本熟悉和回憶起來了。那麼怎麼解決我這個問題呢:
1.cell原本就不是不少,能夠不重用cell,每一個cell都是初始化出來的。
//static NSString *cellIdentifier = @"UserInfoCell";
NSString *cellIdentifier = [NSString stringWithFormat:@"Cell%d%d", [indexPath section], [indexPath row]];//每一個cell的identifier都不一樣,因此不會重用
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier];
}
2.採用重用,重取出來的cell是有可能已經捆綁過數據或者加過子視圖的,因此,若是有必要,要清除數據(好比textlabel的text)和remove掉add過的子視圖。
static NSString *cellIdentifier = @"UserInfoCell";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier];
}
else
{
for (UIView* view in cell.contentView.subviews)
{
[view removeFromSuperview];
}
}
運行一下,發現問題解決了。