UIView中的layoutSubviews和drawRect

转载自简书:http://www.jianshu.com/p/438bcf8e3e53

附加:layoutSubviews方法:

如果你想强制更新布局,你可以调用setNeedsLayout方法;如果你想立即更新你的views,你需要调用layoutIfNeeded方法

正文

UIView里暴露了以下接口,换做之前我是不会管的,基本不会用到;最近由于用到一个第三方代码,需要对其进行改造,不了解清楚的话,感觉是如鲠在喉,化了半天时间做个调研。

我以前做过功能手机上的开发,那个年代没有智能操作系统,每家公司都做自己的操作系统,其中重要且复杂的一块就是关于屏幕上绘制

1,想象一下,一个uiviewcontroller是盖在另一个uiviewcontroller至少,或者一个view遮盖了另外一个view,也许只遮盖了部分,那么被遮盖部分是无需显示到屏幕上的;再复杂一点,上层的view设置了半透明,那么这时下面的view也需要透一点点出来;

2,省电以及效率:显示在屏幕的东西其实都是系统通过一个个写像素点,这是个耗电,耗时间的事;为了省电,除非有视图变化,系统不会刷新页面;如果不是要求实时性特高,系统还会降低刷新频率;省电并提高效率的另一个途径就是只刷新屏幕中需要刷新的部分。

先介绍到此,我们看看与uiview绘制相关的接口如下,分为三组,它们之间相互独立,又相互关联

第一组:布局/定位相关:

(void)setNeedsLayout:在receiver标上一个需要被重新布局的标记,在系统runloop的下一个周期自动调用layoutSubviews。

- (void)layoutIfNeeded:方法如其名,UIKit会判断该receiver是否需要layout.根据Apple官方文档,layoutIfNeeded方法应该是这样的layoutIfNeeded遍历的不是superview链,应该是subviews链。

- (void)layoutSubviews:这是核心函数,最终的目的就是调用该函数,开发者不能调用直接调用该函数,但可以重写该函数,来加入些自己的代码。该函数只会进行位置,视图大小的数字计算,并不会引起屏幕的绘制。

第二组:显示相关:

- (void)setNeedsDisplay:在receiver标上一个需要被重新绘图的标记,在下一个draw周期自动重绘,iphone device的刷新频率是60hz,也就是1/60秒后重绘;

- (void)setNeedsDisplayInRect:(CGRect)rect:不但设置了flag,而且详细规定了需要刷新的区域。

- (void)drawRect:(CGRect)rect:这是核心函数,最终导致显示到屏幕上。开发人员不可以直接调用该函数,只能重写该函数,额外做一些我们想做的事。是对receiver的重绘,能获得context。

第三组:约束相关

setNeedsUpdateConstraints:当一个自定义view的某个属性发生改变,并且可能影响到constraint时,需要调用此方法去标记constraints需要在未来的某个点更新,系统然后调用updateConstraints.

needsUpdateConstraints:constraint-based layout

system使用此返回值去决定是否需要调用updateConstraints作为正常布局过程的一部分。

updateConstraintsIfNeeded:立即触发约束更新,自动更新布局。

updateConstraints:自定义view应该重写此方法在其中建立constraints.注意:要在实现在最后调用[superupdateConstraints]

然后再来说说它们的关系,实际上这是一个view从计算大小,布局,到显示到屏幕上的过程,遵循一定的顺序:

布局过程

Auto layout在view显示之前,多引入了两个步骤:updating constraints和laying out views。每一个步骤都依赖于上一个。display依赖layout,而layout依赖updating constraints,所以顺序是updating

constraints->layout->display;

第一步:updating

constraints,被称为测量阶段,其从下向上(from subview to super view),为下一步layout准备信息。可以通过调用方法setNeedUpdateConstraints去触发此步。constraints的改变也会自动的触发此步。但是,当你自定义view的时候,如果一些改变可能会影响到布局的时候,通常需要自己去通知Auto layout,updateConstraintsIfNeeded。

自定义view的话,通常可以重写updateConstraints方法,在其中可以添加view需要的局部的contraints。

第二步:layout,其从上向下(from super view to subview),此步主要应用上一步的信息去设置view的center和bounds。可以通过调用setNeedsLayout去触发此步骤,此方法不会立即应用layout。如果想要系统立即的更新layout,可以调用layoutIfNeeded。另外,自定义view可以重写方法layoutSubViews来在layout的工程中得到更多的定制化效果。

第三步:display,此步时把view渲染到屏幕上,它与你是否使用Auto layout无关,其操作是从上向下(from super view to subview),通过调用setNeedsDisplay触发,

因为每一步都依赖前一步,因此一个display可能会触发layout,当有任何layout没有被处理的时候,同理,layout可能会触发updating constraints,当constraint system更新改变的时候。

需要注意的是,这三步不是单向的,constraint-based

layout是一个迭代的过程,layout过程中,可能去改变constraints,有一次触发updating constraints,进行一轮layout过程。

注意:如果你每一次调用自定义layoutSubviews都会导致另一个布局传递,那么你将会陷入一个无限循环中。

不能直接调用layoutSubviews和drawRect的原因:

假设我们采用的是直接调用drawRect的机制,先考虑这样一个问题,,现在有两个UIViewController A和B,A为当前view的viewController,如果此时在A中调用[B.view drawRect],这样B的view无论如何都会调用drawRect的方法重新绘制一遍,这样问题就出来了,有必要吗,毕竟现在显示的是A中的view!B重新绘制一遍就调用了drawRect中的方法,完全是在浪费系统资源啊,而通过setNeedsDisplay,ios就会很聪明的判断出 不需要调用drawRect的方法,这样就避免了系统的无用操作!同理,setNeedsLayout也采用了同样的机制来!

何时会调用layoutSubviews,何时需要重写layoutSubviews:

开发者何时需要重写layoutSubviews的问题,Apple是这么解释的:如果自动适配(autosizing),或自动布局(autolayout)不能满足你的要求的时候,就需要重写了。假设一个屏幕从竖屏变为横屏,本来图文单元是图片在上,文字在下面,可能就变为了左右结构;这个是无论自动适配还是自动布局都解决不了,就只能重写layoutSubviews函数了。

但有时候它会被莫名奇妙的调用,根据国外社区一个人帖子,做了总结性翻译。layoutSubviews在以下情况下会被调用:

1、init初始化不会触发layoutSubviews

2、addSubview会触发layoutSubviews

3、设置view的Frame会触发layoutSubviews,当然前提是frame的值设置前后发生了变化

4、滚动一个UIScrollView会触发layoutSubviews

5、旋转Screen会触发父UIView上的layoutSubviews事件

6、改变一个UIView大小的时候也会触发父UIView上的layoutSubviews事件

个人不足:因为在日常工作中我从没需要重写过layoutSubviews和drawRect,最复杂的情况无非用几个view复合在一起就能解决,或者找第三方的代码,所以对实际中为什么要自己调用和重写函数的原因还是不清楚。但目前的这点知识至少可以帮我理解第三方代码了。

此外,这是我在改写的代码,大家可以下载,针对示例进行研究https://github.com/HeshamMegid/HMSegmentedControl

文/践行番茄工作法的攻城狮(简书作者)

原文链接:http://www.jianshu.com/p/438bcf8e3e53

著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容