iOS setNeedsLayout、layoutIfNeeded、setNeedsDisplay

本文源自本人的学习记录整理与理解,其中参考阅读了部分优秀的博客和书籍,尽量以通俗简单的语句转述。引用到的地方如有遗漏或未能一一列举原文出处还望见谅与指出,另文章内容如有不妥之处还望指教,万分感谢。

  • setNeedsLayoutlayoutIfNeeded 被定义在UIView(UIViewHierarchy)分类中的方法,都用于当我们修改了对某一个控件的布局,更新布局的方法,只是时机不同

通过监听runloop状态来区分两者的不同


-  (void)subViewAction {
    
    CFRunLoopObserverRef observerRef = CFRunLoopObserverCreateWithHandler(
                                                                          CFAllocatorGetDefault(),
                                                                          kCFRunLoopAllActivities,
                                                                          YES,
                                                                          0,
                                                                          ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
            switch (activity) {
                case kCFRunLoopEntry:
                    NSLog(@"runloop启动");
                    break;
                case kCFRunLoopBeforeTimers:
                    NSLog(@"runloop即将处理timer事件");
                    break;
                case kCFRunLoopBeforeSources:
                    NSLog(@"runloop即将处理sources事件");
                    break;
                case kCFRunLoopBeforeWaiting:
                    NSLog(@"runloop即将进入休眠");
                    break;
                case kCFRunLoopAfterWaiting:
                    NSLog(@"runloop被唤醒");
                    break;
                case kCFRunLoopExit:
                    NSLog(@"runloop退出");
                    break;
                default:
                    break;
            }
        });
    CFRunLoopAddObserver(CFRunLoopGetCurrent(), observerRef, kCFRunLoopDefaultMode);

    CGFloat height = self.subLabel.frame.size.height + 10;
    self.subLabel.frame = CGRectMake(0, 40, 200, height);
//    [self setNeedsLayout];
//    [self layoutIfNeeded];
    NSLog(@"马上回来");

}

- (void)layoutSubviews {
    [super layoutSubviews];
    NSLog(@"来到 layoutSubviews");
}

  • 场景一 : 两个方法都不用,直接修改控件高度;输出结果:
2021-02-10  马上回来
2021-02-10  runloop即将处理timer事件
2021-02-10  runloop即将处理sources事件
2021-02-10  runloop即将进入休眠
2021-02-10  来到 layoutSubviews
如果使用Masonry 布局 结果相同

说明不做处理时,runloop在即将休眠时更新控件的布局

  • 场景二:setNeedsLayout 输出结果:
2021-02-10  马上回来
2021-02-10  runloop即将处理timer事件
2021-02-10  runloop即将处理sources事件
2021-02-10  runloop即将进入休眠
2021-02-10  来到 layoutSubviews
如果使用Masonry 布局  layoutSubviews 会调两次

和不做处理一样,runloop在即将休眠时更新控件的布局;

  • 场景三: layoutIfNeeded 输出结果:
2021-02-10  来到 layoutSubviews
2021-02-10  马上回来
2021-02-10  runloop即将处理timer事件
2021-02-10  runloop即将处理sources事件
2021-02-10  runloop即将进入休眠
如果使用Masonry 布局 结果相同

立即更新布局

看看官方解释

  • setNeedsLayout :
Call this method on your application’s main thread when you want to adjust the layout of a view’s subviews. 
This method makes a note of the request and returns immediately. 
Because this method does not force an immediate update, but instead waits for the next update cycle, 
you can use it to invalidate the layout of multiple views before any of those views are updated. 
This behavior allows you to consolidate all of your layout updates to one update cycle, which is usually better for performance.

如果要调整视图子视图的布局,请在应用程序的主线程上调用此方法。
此方法记录请求并立即返回。
因为此方法不强制立即更新,而是等待下一个更新周期 
所以可以使用它在更新任何视图之前使多个视图的布局无效
此行为允许您将所有布局更新合并到一个更新周期,这通常会提高性能。

layoutIfNeeded:

Use this method to force the view to update its layout immediately. 
When using Auto Layout, the layout engine updates the position of views as needed to satisfy changes in constraints. 
Using the view that receives the message as the root view, this method lays out the view subtree starting at the root. 
If no layout updates are pending, this method exits without modifying the layout or calling any layout-related callbacks.

使用此方法可强制视图立即更新其布局
使用“自动布局”时,布局引擎会根据需要更新视图的位置,以满足约束的更改
使用接收消息的视图作为根视图,此方法从根开始布置视图子树
如果没有布局更新挂起,则此方法将退出,而不修改布局或调用任何与布局相关的回调

结论:

  • setNeedsLayout : 方法记录更新请求并立即返回; 不强制立即更新,而是等待下一个更新周期;
    应用场景:适用于所有布局更新合并到一个更新周期,这通常会提高性能;

  • layoutIfNeeded :可强制视图立即更新其布局,使用“自动布局”时,布局引擎会根据需要更新视图的位置,以满足约束的更改;但会以接收消息的视图作为根视图,从根视图开始布局视图树;

区别:

  • 触发layoutSubviews的时机不同,前者会在runloop 即将休眠时触发,且一定会触发;后者会立即触发,但前提是控件的布局发生了改变,如果没改变就不会触发

这里的改变:是指用frame布局 width、 height; x、y轴的值改变无效果

Masonry 只要修改布局就会触发layoutSubviews,不修改不会

setNeedsDisplay 是一个容易和以上两个方法混淆的方法

setNeedsDisplay被定义在 UIView(UIViewRendering) 分类中,
Rendering是渲染的意思,可以理解是绘制
Hierarchy是层级、层次的意思,可以理解是布局

setNeedsDisplay的官方解释:

Marks the receiver’s entire bounds rectangle as needing to be redrawn.

You can use this method or the setNeedsDisplayInRect: to notify the system that your view’s contents need to be redrawn. This method makes a note of the request and returns immediately. The view is not actually redrawn until the next drawing cycle, at which point all invalidated views are updated.

可以使用此方法或setNeedsDisplayInRect:通知系统需要重新绘制视图的内容。
此方法记录请求并立即返回。
在下一个绘图周期之前,视图实际上不会重新绘制,此时所有无效的视图都会更新。

You should use this method to request that a view be redrawn only when the content or appearance of the view change.
 If you simply change the geometry of the view, the view is typically not redrawn. 
Instead, its existing content is adjusted based on the value in the view’s contentMode property.
Redisplaying the existing content improves performance by avoiding the need to redraw content that has not changed.

注意

如果视图由caeAglayer对象支持,则此方法无效。它仅适用于使用本机绘图技术(如UIKit和核心图形)渲染其内容的视图。

应该使用此方法请求仅当视图的内容或外观更改时才重新绘制视图。如果仅更改视图的几何图形,则通常不会重新绘制视图。而是根据视图的contentMode属性中的值调整其现有内容。重新显示现有内容可以避免重新绘制未更改的内容,从而提高性能。

名词解释:CAEAGLayer (用OpenGL ES绘制的层)

结论:

  • 记录请求并立即返回,在下一个绘图周期之前,视图实际上不会重新绘制;
  • 使用此方法请求仅当视图的内容或外观更改时才重新绘制视图, 仅更改视图几何图形不回重绘;
  • 应用场景:仅适用于使用本机绘图技术(如UIKit和核心图形)渲染其内容的视图

尝试在修改控件的布局后调用setNeedsDisplay 会发生什么 ?

 [self setNeedsDisplay];
 马上回来
 runloop即将处理timer事件
 runloop即将处理sources事件
 runloop即将进入休眠
 来到 layoutSubviews

神奇的发现竟然和调用setNeedsLayout或什么都不做一样;哈哈哈 说明他只关心渲染和绘制

上面的说法都是建立在子控件的修改后还在父控件上,如果已经超出父控件的显示区域那就一定会触发

layoutSubviews的解释:

The default implementation of this method does nothing on iOS 5.1 and earlier. Otherwise, the default implementation uses any constraints you have set to determine the size and position of any subviews.
Subclasses can override this method as needed to perform more precise layout of their subviews. You should override this method only if the autoresizing and constraint-based behaviors of the subviews do not offer the behavior you want. You can use your implementation to set the frame rectangles of your subviews directly.
You should not call this method directly. If you want to force a layout update, call the setNeedsLayout method instead to do so prior to the next drawing update. If you want to update the layout of your views immediately, call the layoutIfNeeded method.

此方法的默认实现在iOS5.1及更早版本上不起任何作用。否则,默认实现将使用您设置的任何约束来确定任何子视图的大小和位置。

子类可以根据需要重写此方法,以便对其子视图执行更精确的布局。仅当子视图的自动调整大小和基于约束的行为不提供所需的行为时,才应重写此方法。可以使用实现直接设置子视图的框架矩形。

不应直接调用此方法。如果要强制布局更新,请在下次图形更新之前调用setNeedsLayout方法。如果要立即更新视图的布局,请调用layoutIfNeeded方法。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容