iOS回顾笔记(07) -- UITableView的使用和性能优化

iOS回顾笔记(07) -- UITableView的使用和性能优化

如果问iOS中最重要的最常用的UI控件是什么,我觉得UITableView当之无愧!似乎所有常规APP都使用到了UITableView。下面就讲一讲UITableView的常用知识和使用原理及性能优化!

1.简介

  • UITableView故名思议是一种表格控件,是iOS应用中常用的表格控件。常见UITableView如图:

  • UITableView继承于UIScrollview,因此它默认支持垂直滚动(只支持垂直滚动)
  • UITableView性能极佳,它可以出色的完成我们工作中的很多功能。
  • UITableView一共有两种样式:UITableViewStylePlain 和 UITableViewStyleGroup

效果分别如图:

UITableViewStylePlain:一种平铺的效果,并且分组组头默认有停靠效果;


UITableViewStyleGroup:一种组间分离的效果,每组之间间距较明显,有明显分组迹象。

2.数据源

说完了UITableView的简介,下面来说说他它是如何展示数据的。

  1. UITableView要展示数据必须设置数据源,没有数据源(DataSource)它就是一个空壳。
  2. UITableView会调用数据源方法来查询一共有多少行数据以及当前显示什么样的数据。
  3. 凡是遵守 UITableViewDelegate的OC对象都可以做UITableView的数据源

UITableView和数据源的关系如下

数据源方法的调用过程

  1. 调用如下方法,查询一共有多少组数据

    - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView; // Default is 1 if not implemented
    
  2. 调用如下方法,查询每一组一共有多少行数据
    - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section;
    
  3. 调用如下方法,查询每一行显示什么样的内容
    - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath;
    

UITableView还有一些其他的数据源方法如下

/**
 *  返回组头标题
 */
- (nullable NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section;
/**
 *  返回尾标题
 */
- (nullable NSString *)tableView:(UITableView *)tableView titleForFooterInSection:(NSInteger)section;

···

3.UITableViewCell的复用机制

3.1. 返回UITableViewCell的数据源方法的调用

最简单的返回UITableViewCell的方法如下

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    UITableViewCell *cell = [[UITableViewCell alloc] init];
    cell.textLabel.text = [NSString stringWithFormat:@"第 %zd 行",indexPath.row];

    return cell;
}

但是实际使用过程中这样是非常耗费性能的,因为当用户滑动UITableView的时候,上面方法会疯狂的调用,疯狂的创建UITableViewCell。

苹果对这种情况做了一些性能优化,UITableViewCell在每次创建的时候只会创建当前UI页面上可见的Cell。当Cell滑动到屏幕外面,当前Cell会被放入到缓存池中,并且可以给Cell设置一个复用标识,当下次使用Cell的时候可以先到缓存池中查找,如果有对应复用标识的Cell就直接拿出来使用,并重新给内容赋值。

所以根据苹果复用Cell的思路我们在调用返回cell的方法思路应该如下:

  • 1.先从缓存池中查找看有没有可以复用的cell
  • 2.如果有就直接用,如果没有在根据 复用标识 创建
  • 3.创建完成之后对原来数据进行覆盖(防止复用的cell展示异常数据)

所以上面方法从提高性能角度考虑,应该重新写为下面这样:

3.2. UITableView性能优化--Cell复用方式1

/**
 *  什么时候调用:每当有一个cell进入视野范围的时候调用
 */
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    // 0.创建cell复用标识
    static NSString *reuseIdentifier = @"cell";

    // 1.根据复用标识在复用池中进行查找
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:reuseIdentifier];
    // 2.判断如果复用池cell为nil就根据复用标识自己创建
    if (cell == nil) {

        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:reuseIdentifier];
    }

    // 3.拿到cell进行数据覆盖
    cell.textLabel.text = [NSString stringWithFormat:@"第 %zd 行",indexPath.row];

    return cell;
}

3.2. UITableView性能优化--Cell复用方式2

我们可以手动给tableview注册对应标识的cell,保证从缓存池中能加载到对应标识的cell。

通过代码注册

// 注册cell
static NSString *reuseIdentifier = @"cell";
[tableview registerClass:[UITableViewCell class] forCellReuseIdentifier:reuseIdentifier];

通过Xib注册

// 注册cell
static NSString *reuseIdentifier = @"cell"; // 标识可以在Xib中创建
    [tableview registerNib:[UINib nibWithNibName:@"对应的Xib名称" bundle:nil] forCellReuseIdentifier:reuseIdentifier];

在数据源方法中返回cell

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    // 1.根据复用标识在复用池中进行查找
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:reuseIdentifier];

    // 2.拿到cell进行数据覆盖
    cell.textLabel.text = [NSString stringWithFormat:@"第 %zd 行",indexPath.row];

    return cell;
}

3.3 UITableView性能优化--Cell复用方式3

第三种方式是通过UITableViewController在StoryBoard中创建,见下面:5.UITableViewController

4.UITableView的代理方法

有了数据源方法,tableview就可以正常展示数据。有了对应的代理方法就可以正常监听tableview的事件操作。

下面列举一些常用代理方法:

/**
 返回行高,可根据不同行返回不同高

 @param tableView tableview
 @param indexPath 位置
 @return 行高
 */
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath;

/**
 返回估计行高,此方法不进行真实计算,课改变Cell计算高度方法的调用顺序,

 @param tableView tableview
 @param indexPath 位置
 @return 行高
 */
- (CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath;

/**
 cell的点击事件处理

 @param tableView tableview
 @param indexPath 位置
 */
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath;

/**
 cell将要被选中 -- 先需取消选中某个cell才能正常选中新的cell
 */
- (nullable NSIndexPath *)tableView:(UITableView *)tableView willSelectRowAtIndexPath:(NSIndexPath *)indexPath;
/**
 cell将要被取消选中
 */
- (nullable NSIndexPath *)tableView:(UITableView *)tableView willDeselectRowAtIndexPath:(NSIndexPath *)indexPath 

···

5.UITableViewController

UITableViewController继承自UIViewController。可以方便创建有tableview的控制器

  • UITableViewController默认拥有tableview。在UITableViewController中 self.tableView 就是 self.view
  • UITableViewController默认继承 UITableViewDelegate和UITableViewDataSource,所有可以直接实现对应的数据源方法和代理方法即可。
  • 如果在StoryBoard中错误将UIViewController当做UITableViewController使用会报:加载不到UITableview的错误
  • 在StoryBoard中创建UITableViewController并加载cell步骤如下:
    1. 拖一个UITableViewController出来
    2. 设置initial view controller(程序第一个加载项)
    3. 修改对应的类型为自己创建的VC(如果有)
    4. 设置Dynamic Prototypes动态类型的内容,下面数量至少为1
    5. 设置cell的复用标识identfier
    6. 在代码中直接加载对应cell即可(无需判空/创建)
     - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
    {
    // 1.会默认去缓存池中进行查找,如果没有自动去StoryBaord中找
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:reuseIdentifier];
    
    // 2.拿到cell进行数据覆盖
    cell.textLabel.text = [NSString stringWithFormat:@"第 %zd 行",indexPath.row];
    
    return cell;
    }
    
    

    通过StoryBoard创建Cell更加方便

小结

  • UITableView是iOS开发中常用的UI控件
  • UITableView需要实现数据源方法,来确定自己展示什么内容,没有数据源的UITableview只是一个空架子
  • UITableView可以通过复用UITableViewCell来实现性能优化
  • Cell复用的方式有三种(如上)
  • UITableViewController来实现带有tableview的页面更方便

html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,big,cite,code,del,dfn,em,img,ins,kbd,q,s,samp,small,strike,strong,sub,sup,tt,var,b,u,i,center,dl,dt,dd,ol,ul,li,fieldset,form,label,legend,table,caption,tbody,tfoot,thead,tr,th,td,article,aside,canvas,details,embed,figure,figcaption,footer,header,hgroup,menu,nav,output,ruby,section,summary,time,mark,audio,video { margin: 0; padding: 0; border: 0; font: inherit; font-size: 100%; vertical-align: baseline }
html { line-height: 1 }
ol,ul { list-style: none }
table { border-collapse: collapse; border-spacing: 0; margin-top: 0; margin-bottom: 0.8em }
caption,th,td { text-align: left; font-weight: normal; vertical-align: middle }
q,blockquote { quotes: none }
q::before,q::after,blockquote::before,blockquote::after { content: none }
a img { border: none }
article,aside,details,figcaption,figure,footer,header,hgroup,menu,nav,section,summary { display: block }
a { color: #1863a1 }
a:visited { color: #751590 }
a:focus { color: #0181eb }
a:hover { color: #0181eb }
a:active { color: #01579f }
aside.sidebar a { color: #222 }
aside.sidebar a:focus { color: #0181eb }
aside.sidebar a:hover { color: #0181eb }
aside.sidebar a:active { color: #01579f }
a { }
body,h1,h2,h3,h4,h5,h6,footer { font-family: "PT Sans", "Helvetica Neue", "Optima", "Hiragino Sans GB", sans-serif }
body { line-height: 1.5em; color: #222 -webkit-text-size-adjust:none; min-width: 200px; max-width: 760px; margin: 0 auto; padding: 1em }
pre,code,tt,p code,li code { font-family: Menlo, Monaco, "Andale Mono", "lucida console", "Courier New", monospace }
h1 { font-size: 2.2em; line-height: 1.2em }
h1,h2,h3,h4,h5,h6 { margin-bottom: 1em; font-weight: bold }
h2,section h1 { font-size: 1.5em }
h3,section h2,section section h1 { font-size: 1.3em }
h4,section h3,section section h2,section section section h1 { font-size: 1em }
h5,section h4,section section h3 { font-size: .9em }
h6,section h5,section section h4,section section section h3 { font-size: .8em }
.markdown-body { padding: 0px 4px }
.markdown-body h1 { position: relative; padding-top: 1em; padding-bottom: 0.2em; margin-bottom: 1em; background: url("") bottom left repeat-x }
.markdown-body h1 a { text-decoration: none }
.markdown-body h1 a:hover { text-decoration: underline }
.markdown-body h2 { padding-top: 0.8em; padding-bottom: 0.2em; background: url("") bottom left repeat-x }
.markdown-body h2:first-child,.markdown-body header+h2 { padding-top: 4px }
.markdown-body h2:first-child,.markdown-body header+h2 { background: none }
p,.markdown-body blockquote,ul,ol { margin-bottom: 0.8em; margin-top: 0.8em }
ul { list-style-type: disc }
ul ul { list-style-type: circle; margin-bottom: 0px }
ul ul ul { list-style-type: square; margin-bottom: 0px }
ol { list-style-type: decimal }
ol ol { list-style-type: lower-alpha; margin-bottom: 0px }
ol ol ol { list-style-type: lower-roman; margin-bottom: 0px }
ul,ul ul,ul ol,ol,ol ul,ol ol { margin-left: 1.3em }
ul ul,ul ol,ol ul,ol ol { margin-bottom: 0em }
strong { font-weight: bold }
em { font-style: italic }
sup,sub { font-size: 0.75em; position: relative; display: inline-block; padding: 0 .2em; line-height: .8em }
sup { top: -.5em }
sub { bottom: -.5em }
a[rev="footnote"] { font-size: .75em; padding: 0 .3em; line-height: 1 }
q { font-style: italic }
q::before { content: "“" }
q::after { content: "”" }
em,dfn { font-style: italic }
strong,dfn { font-weight: bold }
del,s { text-decoration: line-through }
abbr,acronym { border-bottom: 1px dotted; cursor: help }
small { font-size: .8em }
big { font-size: 1.2em }
.markdown-body hr { height: 0; margin: 15px 0; overflow: hidden; background: transparent; border: 0; border-bottom: 1px solid #ddd }
.markdown-body hr::before { display: table; content: "" }
.markdown-body hr::after { display: table; clear: both; content: "" }
.markdown-body table { display: block; width: 100%; overflow: auto }
.markdown-body table th { font-weight: bold }
.markdown-body table th,.markdown-body table td { padding: 6px 13px; border: 1px solid #ddd }
.markdown-body table tr { background-color: #fff; border-top: 1px solid #ccc }
.markdown-body table tr:nth-child(2n) { background-color: #f8f8f8 }
.markdown-body blockquote { font-style: italic; position: relative; font-size: 1.2em; line-height: 1.5em; padding-left: 1em; border-left: 4px solid rgba(170,170,170,0.5) }
.markdown-body blockquote cite { font-style: italic }
.markdown-body blockquote cite a { color: #aaa !important }
.markdown-body blockquote cite::before { content: "—"; padding-right: .3em; padding-left: .3em; color: #aaa }
.markdown-body a { white-space: pre-wrap }
body>header { font-size: 1em; padding-top: 1.5em; padding-bottom: 1.5em }
.markdown-body { overflow: hidden }
.markdown-body>div,.markdown-body>article { width: 100% }
aside.sidebar { float: none; padding: 0 18px 1px; background-color: #f7f7f7; border-top: 1px solid #e0e0e0 }
.flex-content,article img,article video,article .flash-video,aside.sidebar img { max-width: 100%; height: auto }
.basic-alignment.left,article img.left,article video.left,article .left.flash-video,aside.sidebar img.left { float: left; margin-right: 1.5em }
.basic-alignment.right,article img.right,article video.right,article .right.flash-video,aside.sidebar img.right { float: right; margin-left: 1.5em }
.basic-alignment.center,article img.center,article video.center,article .center.flash-video,aside.sidebar img.center { display: block; margin: 0 auto 1.5em }
.basic-alignment.left,article img.left,article video.left,article .left.flash-video,aside.sidebar img.left,.basic-alignment.right,article img.right,article video.right,article .right.flash-video,aside.sidebar img.right { margin-bottom: .8em }
.toggle-sidebar,.no-sidebar .toggle-sidebar { display: none }
.markdown-body img,.markdown-body video,.markdown-body .flash-video { border: #fff 0.5em solid }
.markdown-body img,.markdown-body video { max-width: 100% }
.markdown-body video,.markdown-body .flash-video { margin: 0 auto 1.5em }
.markdown-body video { display: block; width: 100% }
.markdown-body .flash-video>div { position: relative; display: block; padding-bottom: 56.25%; padding-top: 1px; height: 0; overflow: hidden }
.markdown-body .flash-video>div iframe,.markdown-body .flash-video>div object,.markdown-body .flash-video>div embed { position: absolute; top: 0; left: 0; width: 100%; height: 100% }
.markdown-body>footer { padding-bottom: 2.5em; margin-top: 2em }
.markdown-body>footer p.meta { margin-bottom: .8em; font-size: .85em; clear: both; overflow: hidden }
body,pre { background: #fdf6e3 url("") top left }
body { background-color: #f8f8f8 }
pre { border: 1px solid #e7dec3; line-height: 1.45em; font-size: 13px; margin-bottom: 2.1em; padding: .8em 1em; color: #586e75; overflow: auto }
.markdown-body code { background: none }
h3.filename+pre { }
p code,li code { display: inline-block; white-space: no-wrap; background: #fff; font-size: .8em; line-height: 1.5em; color: #555; border: 1px solid #ddd; padding: 0 .3em; margin: -1px 0 }
p pre code,li pre code { font-size: 1em !important; background: none; border: none }
.hljs { display: block; padding: 0.5em; background: #fdf6e3; color: #657b83 }
.hljs-comment,.diff .hljs-header,.hljs-doctype,.hljs-pi,.lisp .hljs-string { color: #93a1a1 }
.hljs-keyword,.hljs-winutils,.method,.hljs-addition,.css .hljs-tag,.hljs-request,.hljs-status,.nginx .hljs-title { color: #859900 }
.hljs-number,.hljs-command,.hljs-string,.hljs-tag .hljs-value,.hljs-rule .hljs-value,.hljs-doctag,.tex .hljs-formula,.hljs-regexp,.hljs-hexcolor,.hljs-link_url { color: #2aa198 }
.hljs-title,.hljs-localvars,.hljs-chunk,.hljs-decorator,.hljs-built_in,.hljs-identifier,.vhdl .hljs-literal,.hljs-id,.css .hljs-function,.hljs-name { color: #268bd2 }
.hljs-attribute,.hljs-variable,.lisp .hljs-body,.smalltalk .hljs-number,.hljs-constant,.hljs-class .hljs-title,.hljs-parent,.hljs-type,.hljs-link_reference { color: #b58900 }
.hljs-preprocessor,.hljs-preprocessor .hljs-keyword,.hljs-pragma,.hljs-shebang,.hljs-symbol,.hljs-symbol .hljs-string,.diff .hljs-change,.hljs-special,.hljs-attr_selector,.hljs-subst,.hljs-cdata,.css .hljs-pseudo,.hljs-header { color: #cb4b16 }
.hljs-deletion,.hljs-important { color: #dc322f }
.hljs-link_label { color: #6c71c4 }
.tex .hljs-formula { background: #eee8d5 }

时间: 2024-10-04 18:41:08

iOS回顾笔记(07) -- UITableView的使用和性能优化的相关文章

iOS回顾笔记(08) -- 自定义Cell的类型和创建步骤总结

iOS回顾笔记(08) -- 自定义Cell的类型和创建步骤总结 项目中我们常见的自定义cell主要分为两种 等高cell:如应用列表.功能列表? 非等高cell:如微博列表.QQ聊天页面? 下面对这两类cell的创建方式简单记录各步骤. 等高Cell 等高cell通常有三种创建方式: storyboard自定义cell xib自定义cell 代码创建cell(使用frame计算/使用Autolayout布局) 下面分别记录每种创建步骤: 1. storyboard自定义cell 1.创建一个继

iOS回顾笔记(04) -- UIScrollView的基本使用详解

html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,big,cite,code,del,dfn,em,img,ins,kbd,q,s,samp,small,strike,strong,sub,sup,tt,var,b,u,i,center,dl,dt,dd,ol,ul,li,fieldset,form,label,legend,table,caption

iOS回顾笔记( 01 )

html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,big,cite,code,del,dfn,em,img,ins,kbd,q,s,samp,small,strike,strong,sub,sup,tt,var,b,u,i,center,dl,dt,dd,ol,ul,li,fieldset,form,label,legend,table,caption

《android开发艺术探索》读书笔记(十五)--Android性能优化

接上篇<android开发艺术探索>读书笔记(十四)--JNI和NDK编程 No1: 如果<include>制定了这个id属性,同时被包含的布局文件的根元素也制定了id属性,那么以<include>指定的id属性为准 No2: 绘制优化 1)onDraw中不要创建新的局部对象 2)onDraw方法中不要做耗时的任务 No3: 内存泄露优化 场景一:静态变量导致的内存泄露: 如果静态变量持有了一个Activity,会导致Activity无法及时释放. 解决办法:1使用Ap

Linux性能优化从入门到实战:07 CPU篇:CPU性能优化方法

性能优化方法论 ??动手优化性能之前,需要明确以下三个问题: ??(1)如何评估性能优化的效果? 确定性能的量化指标.测试优化前的性能指标.测试优化后的性能指标. ??量化指标的选择.至少要从应用程序和系统资源这两个维度,分别选择不同的指标:1)应用程序的维度,我们可以用吞吐量和请求延迟来评估应用程序的性能.2)系统资源的维度,我们可以用 CPU 使用率来评估系统的 CPU 使用情况. ??行性能测试注意点:1)避免性能测试工具干扰应用程序的性能:2)避免外部环境的变化影响性能指标的评估. ??

iOS学习笔记之UITableViewController&amp;UITableView

iOS学习笔记之UITableViewController&UITableView 写在前面 上个月末到现在一直都在忙实验室的事情,与导师讨论之后,发现目前在实验室完成的工作还不足以写成毕业论文,因此需要继续思考新的算法.这是一件挺痛苦的事情,特别是在很难找到与自己研究方向相关的文献的时候.也许网格序列水印这个课题本身的研究意义就是有待考证的.尽管如此,还是要努力的思考下去.由于实验室的原因,iOS的学习进度明显受到影响,加之整理文档本身是一件耗费时间和精力的事情,因此才这么久没有写笔记了. M

iOS学习笔记(4) — UITableView的 重用机制

iOS学习笔记(4) — UITableView的 重用机制 UITableView中的cell是动态的,在使用过程中,系统会根据屏幕的高度(480)和每个cell的高度计算屏幕中需要显示的cell的个数.比如,cell高度为90.那么480 / 90 = 5 + 1,也就是说最多有6个cell能显示在屏幕中. 系统会创建1个cel池,无论tableview有多少行都只创建6个cell放在池中.当某行移出屏幕的时候,将这个cell放回在池中:当某行需要显示在屏幕中时,从池中取出一个cell. 重

iOS学习笔记—— UItableView 控件的简单使用

UITableView 可以说是iOS开发中最常用的控件,除了游戏之外,几乎所有的应用中独会出现他的身影. 使用UITableView控件需要遵守两种协议 UITableViewDelegate和 UITableViewDataSource. 常用方法如下: 1.返回(每个分区)表单元个数(行数) - (NSInteger) tableView: (UItableView *) tableVIew numberOfRowsInSection: (NSInteger)section 2.返回表单元

iOS 8:【转】UITableView 性能优化笔记

源地址:http://fann.im/blog/2012/09/11/uitableview-optimization-notes/ Hacking Week 技术总结最后一篇,记一下 UITableView 性能优化需要注意和改进的地方. 网络图片异步加载,SDWebImage. 文字直接 drawInRect/drawAtPoint 绘制,参考 ABTableViewCell,AdvancedTableViewCells. 本地图片也可以直接绘制,或者用 CALayer 来添加显示. cel