在iOS中,要實現表格數據展示,最常用的做法就是使用UITableView
UITableView繼承自UIScrollView,因此支持垂直滾動,而且性能極佳
UITableViewStylePlain
UITableViewStyleGrouped
UITableView需要一個數據源(dataSource)來顯示數據
UITableView會向數據源查詢一共有多少行數據以及每一行顯示什么數據等
沒有設置數據源的UITableView只是個空殼
凡是遵守UITableViewDataSource協議的OC對象,都可以是UITableView的數據源
調用數據源的下面方法得知一共有多少組數據
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView;
調用數據源的下面方法得知每一組有多少行數據
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section;
調用數據源的下面方法得知每一行顯示什么內容
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath;
UITableView的每一行都是一個UITableViewCell,通過dataSource的tableView:cellForRowAtIndexPath:方法來初始化每一行
UITableViewCell內部有個默認的子視圖:contentView,contentView是UITableViewCell所顯示內容的父視圖,可顯示一些輔助指示視圖
輔助指示視圖的作用是顯示一個表示動作的圖標,可以通過設置UITableViewCell的accessoryType來顯示,默認是UITableViewCellAccessoryNone(不顯示輔助指示視圖),其他值如下:
UITableViewCellAccessoryDisclosureIndicator
UITableViewCellAccessoryDetailDisclosureButton
UITableViewCellAccessoryCheckmark
還可以通過cell的accessoryView屬性來自定義輔助指示視圖(比如往右邊放一個開關)
contentView下默認有3個子視圖
其中2個是UILabel(通過UITableViewCell的textLabel和detailTextLabel屬性訪問)
第3個是UIImageView(通過UITableViewCell的imageView屬性訪問)
UITableViewCell還有一個UITableViewCellStyle屬性,用于決定使用contentView的哪些子視圖,以及這些子視圖在contentView中的位置
iOS設備的內存有限,如果用UITableView顯示成千上萬條數據,就需要成千上萬個UITableViewCell對象的話,那將會耗盡iOS設備的內存。要解決該問題,需要重用UITableViewCell對象
重用原理:當滾動列表時,部分UITableViewCell會移出窗口,UITableView會將窗口外的UITableViewCell放入一個對象池中,等待重用。當UITableView要求dataSource返回UITableViewCell時,dataSource會先查看這個對象池,如果池中有未使用的UITableViewCell,dataSource會用新的數據配置這個UITableViewCell,然后返回給UITableView,重新顯示到窗口中,從而避免創建新對象。
還有一個非常重要的問題:有時候需要自定義UITableViewCell(用一個子類繼承UITableViewCell),而且每一行用的不一定是同一種UITableViewCell,所以一個UITableView可能擁有不同類型的UITableViewCell,對象池中也會有很多不同類型的UITableViewCell,那么UITableView在重用UITableViewCell時可能會得到錯誤類型的UITableViewCell
解決方案:UITableViewCell有個NSString *reuseIdentifier屬性,可以在初始化UITableViewCell的時候傳入一個特定的字符串標識來設置reuseIdentifier(一般用UITableViewCell的類名)。當UITableView要求dataSource返回UITableViewCell時,先通過一個字符串標識到對象池中查找對應類型的UITableViewCell對象,如果有,就重用,如果沒有,就傳入這個字符串標識來初始化一個UITableViewCell對象
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath: (NSIndexPath *)indexPath{ // 1.定義一個cell的標識 static NSString *ID = @"mjcell";
// 2.從緩存池中取出cell
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];
// 3.如果緩存池中沒有cell
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID];
}
// 4.設置cell的屬性...
return cell;}
1、新建一個xib文件描述一個view的內部結構(假設叫做TWTgCell.xib)
2、新建一個自定義的類
(自定義類需要繼承自系統自帶的view, 繼承自哪個類, ?取決于xib根對象的Class)
3、新建類的類名最好跟xib的文件名保持一致(比如類名就叫做TWTgCell)
4、將xib中的控件 和 自定義類的.m文件 進行連線
5、提供一個類方法返回一個創建好的自定義view(屏蔽從xib加載的過程)
6、提供一個模型屬性讓外界傳遞模型數據
7、重寫模型屬性的setter方法,在這里將模型數據展示到對應的子控件上面
1、對象A內部發生了一些事情,想通知對象B
2、對象B想監聽對象A內部發生了什么事情
3、對象A想在自己的方法內部調用對象B的某個方法,并且對象A不能對對象B有耦合依賴
4、對象A想傳遞數據給對象B
……
以上情況,結果都一樣:對象B是對象A的代理(delegate)
1、先搞清楚誰是誰的代理(delegate)
2、定義代理協議,協議名稱的命名規范:控件類名 + Delegate
3、定義代理方法
代理方法名都以控件名開頭
代理方法至少有1個參數,將控件本身傳遞出去
設置代理(delegate)對象 (比如myView.delegate = xxxx;)
代理對象遵守協議
代理對象實現協議里面該實現的方法
在恰當的時刻調用代理對象(delegate)的代理方法,通知代理發生了什么事情
(在調用之前判斷代理是否實現了該代理方法)
1、新建一個繼承自UITableViewCell的類
2、重寫initWithStyle:reuseIdentifier:方法
添加所有需要顯示的子控件(不需要設置子控件的數據和frame, ?子控件要添加到contentView中)
進行子控件一次性的屬性設置(有些屬性只需要設置一次, 比如字體/固定的圖片)
3、提供2個模型
數據模型: 存放文字數據/圖片數據
frame模型: 存放數據模型/所有子控件的frame/cell的高度
4、cell擁有一個frame模型(不要直接擁有數據模型)
5、重寫frame模型屬性的setter方法: 在這個方法中設置子控件的顯示數據和frame
6、frame模型數據的初始化已經采取懶加載的方式(每一個cell對應的frame模型數據只加載一次)
新聞熱點
疑難解答