iOS UITableView(列表)

UITableView

UITableView內(nèi)置了兩種樣式:UITableViewStylePlain遗菠,UITableViewStyleGrouped

<UITableViewDataSource,UITableViewDelegate>里的方法:
tableView處理步驟

#pragma mark 1.有多少組
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
#pragma mark 2.第section組頭部控件有多高
- (CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(NSInteger)section
#pragma mark 3.第section組有多少行
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
#pragma mark 4.indexPath這行的cell有多高
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
#pragma mark 5.indexPath這行的cell長(zhǎng)什么樣子
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
#pragma mark 6.第section組頭部顯示什么控件
- (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section

//每當(dāng)有一個(gè)cell進(jìn)入視野屏幕就會(huì)調(diào)用献宫,所以在這個(gè)方法內(nèi)部就需要優(yōu)化贮折。
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
if(cell==nil){
//在這里面做創(chuàng)建的工作欧引。循環(huán)優(yōu)化萍嬉。防止刷新cell進(jìn)入屏幕的時(shí)候重復(fù)的創(chuàng)建
}
}

//當(dāng)調(diào)用reloadData的時(shí)候耐床,會(huì)重新刷新調(diào)用數(shù)據(jù)源內(nèi)所有方法,其他事情都不會(huì)做呀
[self reloadData]

//這個(gè)方法只有在一開(kāi)始有多少條數(shù)據(jù)才會(huì)算多少個(gè)高度澜公,這個(gè)方法只會(huì)調(diào)用一次冕末,但是每次reloadData的時(shí)候也會(huì)調(diào)用
//而且會(huì)一次性算出所有cell的高度萍歉,比如有100條數(shù)據(jù),一次性調(diào)用100次
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath

-(NSArray *)sectionIndexTitlesForTableView:(UITableView *)tableView //右側(cè)索引

-(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath //行點(diǎn)擊事件

NSIndexPath *path = [self.tableView indexPathForSelectedRow]; //獲得被選中的indexPath可以得到section档桃,row

[self.tableView reloadRowsAtIndexPaths:[self.tableView indexPathsForSelectedRows] withRowAnimation:UITableViewRowAnimationNone]; //刷新table指定行的數(shù)據(jù)
[self.tableView reloadData]; //刷新table所有行的數(shù)據(jù)

UITableView常用屬性:

UITableView *tableView = [[UITableView alloc]initWithFrame:CGRectMake(0, 0, 320, 460) style:UITableViewStylePlain]; // 初始化表格
分隔線屬性
tableView.separatorStyle = UITableViewCellSeparatorStyleSingleLine; //UITableViewCellSeparatorStyleNone;
[self.tableView setSeparatorStyle:UITableViewCellSeparatorStyleNone]; //取消分隔線
tableView.separatorColor = [UIColor lightGrayColor];

條目多選
tableView.allowsMultipleSelection = YES;

// 設(shè)置標(biāo)題行高
[_tableView setSectionHeaderHeight:kHeaderHeight];
[_tableView setSectionFooterHeight:0];

// 設(shè)置表格行高
[_tableView setRowHeight:50];

//設(shè)置背景色
self.tableView.backgroundView 優(yōu)先級(jí)高枪孩,如果要設(shè)置backgroundColor的時(shí)候要先把view設(shè)置為nil
self.tableView.backgroundColor

//在tableView的頭部或者尾部添加view,footerView寬度是不用設(shè)置的
xxxView.bounds = CGRectMake(0,0,0,height);
self.tableView.tableFooterView =xxxView;
self.tableView.tableHeaderView =xxxView;

UIButton bt = (UIButton)[self.contentView viewWithTag:i+100];

增加tableview滾動(dòng)區(qū)域
self.tableView.contentInset = UIEdgeInsetsMake(0, 0, xx, 0);
UITableViewCell
//創(chuàng)建UITableViewCell
UITableViewCell *cell = [[UITableViewCell alloc]
initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:nil];

[cell.textLabel setBackgroundColor:[UIColor clearColor]];// 清空標(biāo)簽背景顏色
cell.backgroundView =xx; //設(shè)置背景圖片
cell.backgroundVColor =xx;
cell.selectedBackgroundView = selectedBgView; //設(shè)置選中時(shí)的背景顏色

cell.accessoryView = xxxView; //設(shè)置右邊視圖
[cell setAccessoryType:UITableViewCellAccessoryNone]; //設(shè)置右側(cè)箭頭

[self setSelectionStyle:UITableViewCellSelectionStyleNone]; //選中樣式
cell.selectionStyle = UITableViewCellSelectionStyleBlue;

//設(shè)置cell的高度
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath

contentView下默認(rèn)有3個(gè)子視圖,其中的2個(gè)是UILabel,通過(guò)textLabel和detailTextLabel屬性訪問(wèn)蔑舞,第3個(gè)是UIImageView拒担,通過(guò)imageView屬性訪問(wèn).
UITableViewCellStyleDefault, UITableViewCellStyleValue1, UITableViewCellStyleValue2, UITableViewCellStyleSubtitle

#pragma mark - 重新調(diào)整UITalbleViewCell中的控件布局
- (void)layoutSubviews{
[super layoutSubviews];

}
cell 里面還有一個(gè)contentView
UITableViewCell表格優(yōu)化
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畔咧,時(shí)可能會(huì)得到錯(cuò)誤類型的UITableViewCell那么UITableView在重用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ì)象

/**
單元格優(yōu)化

  1. 標(biāo)示符統(tǒng)一洪添,使用static的目的可以保證表格標(biāo)示符永遠(yuǎn)只有一個(gè)
  2. 首先在緩沖池中找名為"myCell"的單元格對(duì)象
  3. 如果沒(méi)有找到,實(shí)例化一個(gè)新的cell
    **/
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
    static NSString *cellIdentifier = @"myCell";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier];
//使用這種方法不用判斷下面的cell
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier forIndexPath:indexPath];
    if (cell == nil) {        
        cell = [[UITableViewCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:cellIdentifier];
    }
return cell;
}

表格的編輯模式
刪除雀费、插入
- (void)setEditing:(BOOL)editing animated:(BOOL)animated; 開(kāi)啟表格編輯狀態(tài)

- (UITableViewCellEditingStyle)tableView:(UITableView *)tableView editingStyleForRowAtIndexPath:(NSIndexPath *)indexPath{
返回表格編輯編輯樣式干奢。不實(shí)現(xiàn)默認(rèn)都是刪除
return editingStyle : UITableViewCellEditingStyleDelete, UITableViewCellEditingStyleInsert
}

- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath{
//根據(jù)editingStyle處理是刪除還是添加操作
完成刪除、插入操作刷新表格
- (void)insertRowsAtIndexPaths:(NSArray *)indexPaths withRowAnimation:(UITableViewRowAnimation)animation;

-(void)deleteRowsAtIndexPaths:(NSArray *)indexPaths withRowAnimation:(UITableViewRowAnimation)animation;
}

移動(dòng)
- (void)tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)sourceIndexPath toIndexPath:(NSIndexPath *)destinationIndexPath
sourceIndexPath 移動(dòng)的行
destinationIndexPath 目標(biāo)的行

自定義表格行UITableViewCell
storyboard方式創(chuàng)建:
直接拖到UITableView里面設(shè)置UITableViewCell
注意:
1.通過(guò)XIB或者Storyboard自定義單元格時(shí)盏袄,在xib和Storyboard里面需要指定單元格的可重用標(biāo)示符Identifier

2.注意表格的優(yōu)化中的差別
在Storyboard中兩者等效
xxCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
xxCell *cell1 = [tableView dequeueReusableCellWithIdentifier:CellIdentifier forIndexPath:indexPath];

在xib文件中有差別:
第一種情況忿峻,只能在iOS 6以上使用,如果在viewDidLoad注冊(cè)了nib文件辕羽,并且指定了“單元格”的可重用標(biāo)示符逛尚,那么
dequeueReusableCellWithIdentifier:
dequeueReusableCellWithIdentifier:forIndexPath:
方法是等效的。如果在viewDidLoad中注冊(cè)了nib文件刁愿,表格緩沖池中的管理绰寞,有系統(tǒng)接管!

第二種情況,是在iOS 4以上均可以使用滤钱,如果沒(méi)有在viewDidLoad注冊(cè)nib文件蕾管,那么,只能使用
dequeueReusableCellWithIdentifier:并且需要判斷cell沒(méi)有被實(shí)例化菩暗,并做相應(yīng)的處理

在代碼創(chuàng)建中差別:
用代碼創(chuàng)建cell中的處理和nib一樣掰曾,注冊(cè)了cell就有系統(tǒng)接管并且可以用帶forIndexPath的方法,沒(méi)有注冊(cè)就要自己去實(shí)例化cell停团,不能用帶forIndexPath的方法
[tableView registerClass:XxxCell class] forCellReuseIdentifier:@"xxCell"];

xib方式創(chuàng)建:
//注冊(cè)Identifier
- (void)viewDidLoad{
[super viewDidLoad];
/**
注意:以下幾句注冊(cè)XIB的代碼旷坦,一定要在viewDidLoad中!
注冊(cè)XIB文件,獲得根視圖佑稠,并且轉(zhuǎn)換成TableView,為tableView注冊(cè)xib
Identifier名要在xib文件中定義秒梅,并且保持一致
**/
UINib *nib = [UINib nibWithNibName:@"BookCell" bundle:[NSBundle mainBundle]];
UITableView *tableView = (UITableView *)self.view;
[tableView registerNib:nib forCellReuseIdentifier:@"bookCell"];
}

//沒(méi)有注冊(cè)Identifier只能使用下面方法
static NSString *CellIdentifier = @"bookCell";
BookCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
if (cell == nil) {
cell = [[BookCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier];
NSBundle *bundle = [NSBundle mainBundle];
NSArray *array = [bundle loadNibNamed:@"BookCell" owner:nil options:nil];
cell = [array lastObject];
}

代碼方式創(chuàng)建:
1. 建立UITableViewCell的類,繼承UITableViewCell
2. 往cell里面加入view的時(shí)候注意點(diǎn):
//新建的組件放入contentView中
[self.contentView addSubview:xxView];

//設(shè)置圖片拉伸屬性stretch
UIImage *normalImage = [UIImage imageNamed:@"xx.png"];
normalImage = [normalImage stretchableImageWithLeftCapWidth:
normalImage.size.width / 2 topCapHeight:normalImage.size.height / 2];

//在tableView里面viewDiDLoad里面要注冊(cè)cell類
[tableView registerClass:XxxCell class] forCellReuseIdentifier:@"xxCell"];

自定義表格中Header
//自定義表格在這個(gè)方法中定義
-(UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section
-(NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末舌胶,一起剝皮案震驚了整個(gè)濱河市捆蜀,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌幔嫂,老刑警劉巖辆它,帶你破解...
    沈念sama閱讀 207,113評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件履恩,死亡現(xiàn)場(chǎng)離奇詭異锰茉,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)飒筑,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)绽昏,“玉大人协屡,你說(shuō)我怎么就攤上這事∪” “怎么了肤晓?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,340評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)啼县。 經(jīng)常有香客問(wèn)我材原,道長(zhǎng),這世上最難降的妖魔是什么季眷? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,449評(píng)論 1 279
  • 正文 為了忘掉前任余蟹,我火速辦了婚禮,結(jié)果婚禮上子刮,老公的妹妹穿的比我還像新娘威酒。我一直安慰自己窑睁,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布葵孤。 她就那樣靜靜地躺著担钮,像睡著了一般。 火紅的嫁衣襯著肌膚如雪尤仍。 梳的紋絲不亂的頭發(fā)上箫津,一...
    開(kāi)封第一講書(shū)人閱讀 49,166評(píng)論 1 284
  • 那天,我揣著相機(jī)與錄音宰啦,去河邊找鬼苏遥。 笑死,一個(gè)胖子當(dāng)著我的面吹牛赡模,可吹牛的內(nèi)容都是我干的田炭。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼漓柑,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼教硫!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起辆布,我...
    開(kāi)封第一講書(shū)人閱讀 37,105評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤瞬矩,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后谚殊,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體丧鸯,經(jīng)...
    沈念sama閱讀 43,601評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡蛤铜,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評(píng)論 2 325
  • 正文 我和宋清朗相戀三年嫩絮,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片围肥。...
    茶點(diǎn)故事閱讀 38,161評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡剿干,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出穆刻,到底是詐尸還是另有隱情置尔,我是刑警寧澤,帶...
    沈念sama閱讀 33,792評(píng)論 4 323
  • 正文 年R本政府宣布氢伟,位于F島的核電站榜轿,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏朵锣。R本人自食惡果不足惜谬盐,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望诚些。 院中可真熱鬧飞傀,春花似錦皇型、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,352評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至幢痘,卻和暖如春唬格,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背颜说。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,584評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工西轩, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人脑沿。 一個(gè)月前我還...
    沈念sama閱讀 45,618評(píng)論 2 355
  • 正文 我出身青樓藕畔,卻偏偏與公主長(zhǎng)得像贡翘,于是被迫代替她去往敵國(guó)和親坚踩。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評(píng)論 2 344

推薦閱讀更多精彩內(nèi)容