?在眾多移動(dòng)應(yīng)用中,能看到各式各樣的表格數(shù)據(jù)
在iOS中,要實(shí)現(xiàn)表格數(shù)據(jù)展示,最常用的做法就是使用UITableView
UITableView繼承自UIScrollView,因此支持垂直滾動(dòng),而且性能極佳
最常見(jiàn)的兩種方式:
UITableViewStylePlain和UITableViewStyleGrouped
如何展示數(shù)據(jù):
?UITableView需要一個(gè)數(shù)據(jù)源(dataSource)來(lái)顯示數(shù)據(jù)
?UITableView會(huì)向數(shù)據(jù)源查詢一共有多少行數(shù)據(jù)以及每一行顯示什么數(shù)據(jù)等
?沒(méi)有設(shè)置數(shù)據(jù)源的UITableView只是個(gè)空殼
?凡是遵守UITableViewDataSource協(xié)議的OC對(duì)象,都可以是UITableView的數(shù)據(jù)源
1.調(diào)用數(shù)據(jù)源的下面方法得知一共有多少組數(shù)據(jù)
-(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView;
2.調(diào)用數(shù)據(jù)源的下面方法得知每一組有多少行數(shù)據(jù)
-(NSInteger)tableView:(UITableView *)tableView
numberOfRowsInSection:(NSInteger)section;
3.調(diào)用數(shù)據(jù)源的下面方法得知每一行顯示什么內(nèi)容
-(UITableViewCell *)tableView:(UITableView *)tableView
cellForRowAtIndexPath:(NSIndexPath *)indexPath;
tableView的數(shù)據(jù)源必須實(shí)現(xiàn)第二個(gè)方法和第三個(gè)方法,第一個(gè)方法不實(shí)現(xiàn)默認(rèn)就是1組
初始MVC
?MVC是一種設(shè)計(jì)思想,貫穿于整個(gè)iOS開(kāi)發(fā)中,需要積累一定的項(xiàng)目經(jīng)驗(yàn),才能深刻體會(huì)其中的含義和好處
?MVC中的三個(gè)角色
?M:Model,模型數(shù)據(jù)
?V:View,視圖(界面)
?C:Control,控制中心
?MVC的幾個(gè)明顯的特征和體現(xiàn):
?View上面顯示什么東西,取決于Model
?只要Model數(shù)據(jù)改了,View的顯示狀態(tài)會(huì)跟著更改
?Control負(fù)責(zé)初始化Model,并將Model傳遞給View去解析展示
Cell簡(jiǎn)介
?UITableView中的每一行都是一個(gè)UITableViewCell,通過(guò)dataSource的tableView:cellForRowAtIndexPath:方法來(lái)初始化每一行
?UITableViewCell內(nèi)部有個(gè)默認(rèn)的子視圖:contentView,contentView是UITableViewCell所顯示內(nèi)容的父視圖,可顯示一些輔助指示視圖
?輔助指示視圖的作用是顯示一個(gè)表示動(dòng)作的圖標(biāo),可以通過(guò)設(shè)置UITableViewCell的accessoryType來(lái)顯示,默認(rèn)是UITableViewCellAccessoryNone(不顯示輔助指示視圖),
?還可以通過(guò)cell的accessoryView屬性來(lái)自定義輔助指示視圖(比如往右邊放一個(gè)開(kāi)關(guān))
?contentView下默認(rèn)有3個(gè)子視圖
?其中2個(gè)是UILabel(通過(guò)UITableViewCell的textLabel和detailTextLabel屬性訪問(wèn))
?第3個(gè)是UIImageView(通過(guò)UITableViewCell的imageView屬性訪問(wèn))
?UITableViewCell還有一個(gè)UITableViewCellStyle屬性,用于決定使用contentView的哪些子視圖,以及這些子視圖在contentView中的位置
Cell的重用原理
?iOS設(shè)備的內(nèi)存有限,如果用UITableView顯示成千上萬(wàn)條數(shù)據(jù),就需要成千上萬(wàn)個(gè)UITableViewCell對(duì)象的話,那將會(huì)耗盡iOS設(shè)備的內(nèi)存。要解決該問(wèn)題,需要重用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è)非常重要的問(wèn)題:有時(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í)來(lái)設(shè)置reuseIdentifier(一般用UITableViewCell的類名)。當(dāng)UITableView要求dataSource返回UITableViewCell時(shí),先通過(guò)一個(gè)字符串標(biāo)識(shí)到對(duì)象池中查找對(duì)應(yīng)類型的UITableViewCell對(duì)象,如果有,就重用,如果沒(méi)有,就傳入這個(gè)字符串標(biāo)識(shí)來(lái)初始化一個(gè)UITableViewCell對(duì)象
使用xib封裝一個(gè)view的步驟
1.新建一個(gè)xib文件描述一個(gè)view的內(nèi)部結(jié)構(gòu)(假設(shè)叫做HMTgCell.xib)
2.新建一個(gè)自定義的類
(自定義類需要繼承自系統(tǒng)自帶的view, 繼承自哪個(gè)類, ?取決于xib根對(duì)象的Class)
3.新建類的類名最好跟xib的文件名保持一致(比如類名就叫做HMTgCell)
4.將xib中的控件
和自定義類的.m文件進(jìn)行連線
5.提供一個(gè)類方法返回一個(gè)創(chuàng)建好的自定義view(屏蔽從xib加載的過(guò)程)
6.提供一個(gè)模型屬性讓外界傳遞模型數(shù)據(jù)
7.重寫(xiě)模型屬性的setter方法,在這里將模型數(shù)據(jù)展示到對(duì)應(yīng)的子控件上面
Delegate的使用場(chǎng)合
?對(duì)象A內(nèi)部發(fā)生了一些事情,想通知對(duì)象B
?對(duì)象B想監(jiān)聽(tīng)對(duì)象A內(nèi)部發(fā)生了什么事情
?對(duì)象A想在自己的方法內(nèi)部調(diào)用對(duì)象B的某個(gè)方法,并且對(duì)象A不能對(duì)對(duì)象B有耦合依賴
?對(duì)象A想傳遞數(shù)據(jù)給對(duì)象B
?……
以上情況,結(jié)果都一樣:對(duì)象B是對(duì)象A的代理(delegate)
1.先搞清楚誰(shuí)是誰(shuí)的代理(delegate)
2.在委托方的.h定義代理協(xié)議,協(xié)議名稱的命名規(guī)范:控件類名+Delegate
3.定義代理方法
?代理方法一般都用@optional修飾
?代理方法名都以控件名開(kāi)頭
?代理方法至少有1個(gè)參數(shù),將控件本身傳遞出去
?定義代理屬性(@property
(nonatomic,weak) id delegate;)
4.設(shè)置代理(delegate)對(duì)象(比如myView.delegate=xxxx;)
?代理對(duì)象遵守協(xié)議
?代理對(duì)象實(shí)現(xiàn)協(xié)議里面該實(shí)現(xiàn)的方法
5.在恰當(dāng)?shù)臅r(shí)刻調(diào)用代理對(duì)象(delegate)的代理方法,通知代理發(fā)生了什么事情
(在調(diào)用之前判斷代理是否實(shí)現(xiàn)了該代理方法)
通過(guò)代碼自定義cell(cell的高度不一致)
1.新建一個(gè)繼承自UITableViewCell的類
2.重寫(xiě)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.重寫(xiě)frame模型屬性的setter方法: 在這個(gè)方法中設(shè)置子控件要顯示的數(shù)據(jù)和frame
6.frame模型數(shù)據(jù)的初始化已經(jīng)采取懶加載的方式(每一個(gè)cell對(duì)應(yīng)的frame模型數(shù)據(jù)只加載一次)
UITextField
?通過(guò)UITextField的代理方法能夠監(jiān)聽(tīng)鍵盤(pán)最右下角按鈕的點(diǎn)擊
1.成為UITextField的代理
self.textField.delegate = self;
2.遵守UITextFieldDelegate協(xié)議,實(shí)現(xiàn)代理方法
-(BOOL)textFieldShouldReturn:(UITextField*)textField;
?在UITextField左邊放一個(gè)view
self.textField.leftView = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 8, 0)];
self.textField.leftViewMode = UITextFieldViewModeAlways;