Cell簡(jiǎn)介
- UITableView的每一行都是一個(gè)UITableViewCell,通過dataSource的tableView:cellForRowAtIndexPath:方法來初始化每一行
- UITableViewCell內(nèi)部有個(gè)默認(rèn)的子視圖:contentView,contentView是UITableViewCell所顯示內(nèi)容的父視圖,可顯示一些輔助指示視圖
-? 輔助指示視圖的作用是顯示一個(gè)表示動(dòng)作的圖標(biāo),可以通過設(shè)置UITableViewCell的accessoryType來顯示,默認(rèn)是UITableViewCellAccessoryNone(不顯示輔助指示視圖),其他值如下:
- UITableViewCellAccessoryDisclosureIndicator
- UITableViewCellAccessoryDetailDisclosureButton
- UITableViewCellAccessoryCheckmark
- 還可以通過cell的accessoryView屬性來自定義輔助指示視圖(比如往右邊放一個(gè)開關(guān))
Cell的重用原理
- iOS設(shè)備的內(nèi)存有限,如果用UITableView顯示成千上萬條數(shù)據(jù),就需要成千上萬個(gè)UITableViewCell對(duì)象的話,那將會(huì)耗盡iOS設(shè)備的內(nèi)存。要解決該問題,需要重用UITableViewCell對(duì)象
- 重用原理:當(dāng)滾動(dòng)列表時(shí),部分UITableViewCell會(huì)移出窗口,UITableView會(huì)將窗口外的UITableViewCell放入一個(gè)對(duì)象池中,等待重用。當(dāng)UITableView要求dataSource返回UITableViewCell時(shí),dataSource會(huì)先查看這個(gè)對(duì)象池,如果池中有未使用的UITableViewCell,dataSource會(huì)用新的數(shù)據(jù)配置這個(gè)UITableViewCell,然后返回給UITableView,重新顯示到窗口中,從而避免創(chuàng)建新對(duì)象
- 還有一個(gè)非常重要的問題:有時(shí)候需要自定義UITableViewCell(用一個(gè)子類繼承UITableViewCell),而且每一行用的不一定是同一種UITableViewCell,所以一個(gè)UITableView可能擁有不同類型的UITableViewCell,對(duì)象池中也會(huì)有很多不同類型的UITableViewCell,那么UITableView在重用UITableViewCell時(shí)可能會(huì)得到錯(cuò)誤類型的UITableViewCell
- 解決方案:UITableViewCell有個(gè)NSString *reuseIdentifier屬性,可以在初始化UITableViewCell的時(shí)候傳入一個(gè)特定的字符串標(biāo)識(shí)來設(shè)置reuseIdentifier(一般用UITableViewCell的類名)。當(dāng)UITableView要求dataSource返回UITableViewCell時(shí),先通過一個(gè)字符串標(biāo)識(shí)到對(duì)象池中查找對(duì)應(yīng)類型的UITableViewCell對(duì)象,如果有,就重用,如果沒有,就傳入這個(gè)字符串標(biāo)識(shí)來初始化一個(gè)UITableViewCell對(duì)象
Cell的重用代碼
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
// 1.定義一個(gè)cell的標(biāo)識(shí)
static NSString *ID = @”czcell";
// 2.從緩存池中取出cell
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
// 3.如果緩存池中沒有cell
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID];
}
tableView性能優(yōu)化 - cell的循環(huán)利用方式1
// 什么時(shí)候調(diào)用:每當(dāng)有一個(gè)cell進(jìn)入視野范圍內(nèi)就會(huì)調(diào)用
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
// 0.重用標(biāo)識(shí)
// 被static修飾的局部變量:只會(huì)初始化一次,在整個(gè)程序運(yùn)行過程中,只有一份內(nèi)存
static NSString *ID = @"cell";
// 1.先根據(jù)cell的標(biāo)識(shí)去緩存池中查找可循環(huán)利用的cell
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
// 2.如果cell為nil(緩存池找不到對(duì)應(yīng)的cell)
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:ID];
}
// 3.覆蓋數(shù)據(jù)
cell.textLabel.text = [NSString stringWithFormat:@"testdata - %zd", indexPath.row];
return cell;
}
tableView性能優(yōu)化 - cell的循環(huán)利用方式2
- 定義一個(gè)全局變量
// 定義重用標(biāo)識(shí)
NSString *ID = @"cell";
- 注冊(cè)某個(gè)標(biāo)識(shí)對(duì)應(yīng)的cell類型
// 在這個(gè)方法中注冊(cè)cell
- (void)viewDidLoad {
[super viewDidLoad];
// 注冊(cè)某個(gè)標(biāo)識(shí)對(duì)應(yīng)的cell類型
[self.tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:ID];
}
- 在數(shù)據(jù)源方法中返回cell
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
// 1.去緩存池中查找cell
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
// 2.覆蓋數(shù)據(jù)
cell.textLabel.text = [NSString stringWithFormat:@"testdata - %zd", indexPath.row];
return cell;
}
tableView性能優(yōu)化 - cell的循環(huán)利用方式3
在storyboard中設(shè)置UITableView的Dynamic Prototypes Cell
設(shè)置cell的重用標(biāo)識(shí)
在代碼中利用重用標(biāo)識(shí)獲取cell
// 0.重用標(biāo)識(shí)
// 被static修飾的局部變量:只會(huì)初始化一次,在整個(gè)程序運(yùn)行過程中,只有一份內(nèi)存
static NSString *ID = @"cell";
// 1.先根據(jù)cell的標(biāo)識(shí)去緩存池中查找可循環(huán)利用的cell
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
// 2.覆蓋數(shù)據(jù)
cell.textLabel.text = [NSString stringWithFormat:@"cell - %zd", indexPath.row];
return cell;
自定義cell:等高的cell
? ? ? ? ? ? ? storyboard自定義cell ??
?1.創(chuàng)建一個(gè)繼承自UITableViewCell的子類,比如XMGDealCell
2.在storyboard中? ? ? ? ? ?
? ? ? ? ? ? ? ? ? ?往cell里面增加需要用到的子控件
? ? ? ? ? ? ? ? ? ? 設(shè)置cell的重用標(biāo)識(shí)
? ? ? ? ? ? ? ? ? 設(shè)置cell的class為XMGDealCell
3.在控制器中
? ? ? ? ? ? - 利用重用標(biāo)識(shí)找到cell
? ? ? ? ? ? - 給cell傳遞模型數(shù)據(jù)
4.在XMGDealCell中
? ? ? ? - 將storyboard中的子控件連線到類擴(kuò)展中
需要提供一個(gè)模型屬性,重寫模型的set方法,在這個(gè)方法中設(shè)置模型數(shù)據(jù)到子控件上
xib自定義cell
-1.創(chuàng)建一個(gè)繼承自UITableViewCell的子類,比如XMGDealCell
- 2.創(chuàng)建一個(gè)xib文件(文件名建議跟cell的類名一樣),比如XMGDealCell.xib? ? ??
? ? ? - 拖拽一個(gè)UITableViewCell出來? ? ? ?
? ? ? - 修改cell的class為XMGDealCell? ? ? ? ??
? ? ? - **設(shè)置cell的重用標(biāo)識(shí)**? ? ? ??
? ? ? - 往cell中添加需要用到的子控件? ??
- 3.在控制器中? ? ? ? ? ?
? ? ? ? ?- **利用registerNib...方法注冊(cè)xib文件**? ? ? ? ??
? ? ? ?? - 利用重用標(biāo)識(shí)找到cell(如果沒有注冊(cè)xib文件,就需要手動(dòng)去加載xib文件)? ? ? ? ? ??
? ? ? ? - 給cell傳遞模型數(shù)據(jù)
- 4.在XMGDealCell中? ? ? ? ?
? ? ? ?- 將xib中的子控件連線到類擴(kuò)展中? ? ?
?? ? ? - 需要提供一個(gè)模型屬性,重寫模型的set方法,在這個(gè)方法中設(shè)置模型數(shù)據(jù)到子控件上? ? ??
? ? ? ?- 也可以將創(chuàng)建獲得cell的代碼封裝起來(比如cellWithTableView:方法)? ??
? 代碼自定義cell(使用frame)
?- 1.創(chuàng)建一個(gè)繼承自UITableViewCell的子類,比如XMGDealCell ? ? ? ? ?
? ? ? ? ? ? ? ? ? ? ?- 在initWithStyle:reuseIdentifier:方法中
? ? ? ? ? ? ? ? ? ? ?- 添加子控件
? ? ? ? ? ? ? ? ? ? ?- 設(shè)置子控件的初始化屬性(比如文字顏色、字體)
? ? ? ? ? ? ? ? ? ? - 在layoutSubviews方法中設(shè)置子控件的frame
? ? ? ? ? ? ? ? ? ? - 需要提供一個(gè)模型屬性,重寫模型的set方法,在這個(gè)方法中設(shè)置模型數(shù)據(jù)到子控件
? - 2.在控制器中
? ? ? ? ? ? ? ? - 利用registerClass...方法注冊(cè)XMGDealCell類
? ? ? ? ? ? ? ? - 利用重用標(biāo)識(shí)找到cell(如果沒有注冊(cè)類,就需要手動(dòng)創(chuàng)建cell)
? ? ? ? ? ? ? ? - 給cell傳遞模型數(shù)據(jù)
? ? ? ? ? ? ? ?- 也可以將創(chuàng)建獲得cell的代碼封裝起來(比如cellWithTableView:方法)
代碼自定義cell(使用autolayout)
- 1.創(chuàng)建一個(gè)繼承自UITableViewCell的子類,比如XMGDealCell
? ? ?- 在initWithStyle:reuseIdentifier:方法中
? ? ? ? ?- 添加子控件
? ? ? ? ?- 添加子控件的約束(建議使用`Masonry`)
? ? ? ? ?- 設(shè)置子控件的初始化屬性(比如文字顏色、字體)
? ? ?- 需要提供一個(gè)模型屬性,重寫模型的set方法,在這個(gè)方法中設(shè)置模型數(shù)據(jù)到子控件?
- 2.在控制器中
? ?- 利用registerClass...方法注冊(cè)XMGDealCell類
? ?- 利用重用標(biāo)識(shí)找到cell(如果沒有注冊(cè)類,就需要手動(dòng)創(chuàng)建cell)
? ?- 給cell傳遞模型數(shù)據(jù)
? ?- 也可以將創(chuàng)建獲得cell的代碼封裝起來(比如cellWithTableView:方法)
非等高的cell
- xib自定義cell
- storyboard自定義cell
- 代碼自定義cell(frame)
- 代碼自定義cell(Autolayout)
通過代碼自定義cell(cell的高度不一致)
- 1.新建一個(gè)繼承自UITableViewCell的類
- 2.重寫initWithStyle:reuseIdentifier:方法
- 添加所有需要顯示的子控件(不需要設(shè)置子控件的數(shù)據(jù)和frame, 子控件要添加到contentView中)
- 進(jìn)行子控件一次性的屬性設(shè)置(有些屬性只需要設(shè)置一次, 比如字體\固定的圖片)
- 3.提供2個(gè)模型
- 數(shù)據(jù)模型: 存放文字?jǐn)?shù)據(jù)\圖片數(shù)據(jù)
- frame模型: 存放數(shù)據(jù)模型\所有子控件的frame\cell的高度
- 4.cell擁有一個(gè)frame模型(不要直接擁有數(shù)據(jù)模型)
- 5.重寫frame模型屬性的setter方法: 在這個(gè)方法中設(shè)置子控件的顯示數(shù)據(jù)和frame
- 6.frame模型數(shù)據(jù)的初始化已經(jīng)采取懶加載的方式(每一個(gè)cell對(duì)應(yīng)的frame模型數(shù)據(jù)只加載一次)