视图layoutSubviews方法调用时机研究

版本记录

版本号 时间
V1.0 2017.12.10

前言

无论是初学者还是资深的ios开发工程师,相信对视图的layoutSubviews方法都不陌生,这个方法很好用,但是前提是要对其调用机制和时机非常的了解,要不就可能产生很诡异的视图显示问题。接下来我们就一起研究一下该方法的调用时机,希望对大家有所帮助。

方法的调用时机

下面我们就一起来看一下layoutSubviews方法的调用时机。

1. alloc - init方法

下面我们看一下在实力化视图的alloc - init方法掉用后会不会接着调用视图layoutSubviews方法。

下面我们看一下代码。

1. ViewController.m
#import "ViewController.h"
#import "JJTestViewOne.h"

@interface ViewController ()

@end

@implementation ViewController

#pragma mark - Override Base Function

- (void)viewDidLoad
{
    [super viewDidLoad];
    
    JJTestViewOne *viewOne = [[JJTestViewOne alloc] init];
    [self.view addSubview:viewOne];
}

@end
2. JJTestViewOne.m
#import "JJTestViewOne.h"

@implementation JJTestViewOne

#pragma mark - Override Base Function

- (void)layoutSubviews
{
    NSLog(@"layoutSubviews被调用了");
}

@end

下面看输出结果

2017-12-10 09:58:29.674737+0800 JJLayoutSubviews[1351:27183] layoutSubviews被调用了

可见,alloc - init进行实例化的时候系统是会调用layoutSubviews方法的。

我查阅了网上资料,发现有个博客是14年写的,说是调用alloc - init进行实例化不会去调用layoutSubviews方法,但是从上面可以看出来,确实是调用了,不知道是视图的调用机制改变了,还是那位博主写错了,不管一切我们还是以实际测试和代码输出为准,毕竟代码是不会骗人的。

这里还有一个小点,需要注意,我刚才测试实例化的代码时这么写的。

JJTestViewOne *viewOne = [[JJTestViewOne alloc] init];
[self.view addSubview:viewOne];

但是如果像下面这么写

 JJTestViewOne *viewOne = [[JJTestViewOne alloc] initWithFrame:self.view.bounds];
 [self.view addSubview:viewOne];

视图中也重写一个父类方法

- (instancetype)initWithFrame:(CGRect)frame
{
    if (self = [super initWithFrame:frame]) {
        NSLog(@"initWithFrame");
    }
    return self;
}

这时候就会发现系统调用layoutSubviews方法一共调用了两次,看一下输出结果。

2017-12-10 10:08:39.627853+0800 JJLayoutSubviews[1471:33931] initWithFrame
2017-12-10 10:08:43.211734+0800 JJLayoutSubviews[1471:33931] layoutSubviews被调用了
2017-12-10 10:08:43.211992+0800 JJLayoutSubviews[1471:33931] layoutSubviews被调用了

2. addSubview添加子视图时

这个大家应该没什么疑问,添加子视图系统就会调用视图layoutSubviews方法。

我先添加一个子视图,如下代码。

#import "JJTestViewOne.h"

@implementation JJTestViewOne

#pragma mark - Override Base Function

- (instancetype)initWithFrame:(CGRect)frame
{
    if (self = [super initWithFrame:frame]) {
        NSLog(@"initWithFrame");
        [self initUI];
    }
    return self;
}

- (void)layoutSubviews
{
    NSLog(@"layoutSubviews被调用了");
}

#pragma mark - Object Private Function

- (void)initUI
{
    UIView *view = [[UIView alloc] initWithFrame:CGRectMake(100.0, 200.0, 100.0, 100.0)];
    view.backgroundColor = [UIColor yellowColor];
    [self addSubview:view];
}

@end

下面看输出结果

2017-12-10 10:19:07.053563+0800 JJLayoutSubviews[1554:39775] initWithFrame
2017-12-10 10:19:16.117887+0800 JJLayoutSubviews[1554:39775] layoutSubviews被调用了
2017-12-10 10:19:17.266040+0800 JJLayoutSubviews[1554:39775] layoutSubviews被调用了

通过单步调试,可以发现添加子视图[self addSubview:view],确实系统要调用layoutSubviews方法。

3. 更改view的Frame

我下面在VC中touchesBegan方法中进行修改子view的frame方法,如下所示。

#import "ViewController.h"
#import "JJTestViewOne.h"

@interface ViewController ()

@property (nonatomic, strong) JJTestViewOne *viewOne;

@end

@implementation ViewController

#pragma mark - Override Base Function

- (void)viewDidLoad
{
    [super viewDidLoad];
    
    self.viewOne = [[JJTestViewOne alloc] initWithFrame:self.view.bounds];
    self.viewOne.backgroundColor = [UIColor yellowColor];
    [self.view addSubview:self.viewOne];
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    self.viewOne.frame = CGRectMake(100.0, 200.0, 100.0, 100.0);
}

@end

下面看输出

2017-12-10 10:28:44.692919+0800 JJLayoutSubviews[1794:49520] initWithFrame
2017-12-10 10:28:44.696648+0800 JJLayoutSubviews[1794:49520] layoutSubviews被调用了
2017-12-10 10:28:44.696796+0800 JJLayoutSubviews[1794:49520] layoutSubviews被调用了
2017-12-10 10:28:47.146150+0800 JJLayoutSubviews[1794:49520] layoutSubviews被调用了

这里最后一条数据就是touch屏幕后的输出数据,说明改变子视图的frame系统会调用视图的layoutSubviews方法。

2017-12-10 10:28:47.146150+0800 JJLayoutSubviews[1794:49520] layoutSubviews被调用了

这里需要注意,只有设置后的frame与之前相比发生了变化,系统才会去调用layoutSubviews方法。下面我们测试一下。

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
//    self.viewOne.frame = CGRectMake(100.0, 200.0, 100.0, 100.0);
    self.viewOne.frame = self.view.bounds;
}

下面看输出结果

2017-12-10 10:46:14.744995+0800 JJLayoutSubviews[1884:57040] initWithFrame
2017-12-10 10:46:14.748756+0800 JJLayoutSubviews[1884:57040] layoutSubviews被调用了
2017-12-10 10:46:14.748901+0800 JJLayoutSubviews[1884:57040] layoutSubviews被调用了

这里我重新设置一下视图的frame,self.viewOne.frame = self.view.bounds;,相当于frame没有变化,系统就不会调用layoutSubviews方法了。

4. 改变子视图的大小触发父视图上的layoutSubviews方法

下面还是直接看代码。

#import "JJTestViewOne.h"

@interface JJTestViewOne()

@property (nonatomic, strong) UIView *subView;

@end

@implementation JJTestViewOne

#pragma mark - Override Base Function

- (instancetype)initWithFrame:(CGRect)frame
{
    if (self = [super initWithFrame:frame]) {
        NSLog(@"initWithFrame");
        [self initUI];
        UITapGestureRecognizer *tapGesture = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(viewDidTapped)];
        [self addGestureRecognizer:tapGesture];
    }
    return self;
}

- (void)layoutSubviews
{
    NSLog(@"layoutSubviews被调用了");
}

#pragma mark - Object Private Function

- (void)initUI
{
    UIView *view = [[UIView alloc] initWithFrame:CGRectMake(100.0, 200.0, 100.0, 100.0)];
    view.backgroundColor = [UIColor redColor];
    [self addSubview:view];
    self.subView = view;
}
#pragma mark - Action && Notification

- (void)viewDidTapped
{
    self.subView.frame = CGRectMake(100.0, 200.0, 300.0, 300.0);
}

@end

看输出结果

2017-12-10 10:53:35.650724+0800 JJLayoutSubviews[1914:60739] initWithFrame
2017-12-10 10:53:35.654739+0800 JJLayoutSubviews[1914:60739] layoutSubviews被调用了
2017-12-10 10:53:35.654941+0800 JJLayoutSubviews[1914:60739] layoutSubviews被调用了
2017-12-10 10:55:38.448257+0800 JJLayoutSubviews[1914:60739] layoutSubviews被调用了

这里在响应手势方法里面,执行代码self.subView.frame = CGRectMake(100.0, 200.0, 300.0, 300.0);,改变了子视图的frame,就会发现父视图的layoutSubviews调用了。

5. scrollview等滚动视图滚动时调用

下面我们就看一下滚动视图在滚动时候进行调用layoutSubviews方法的情况。

下面还是直接看代码。

- (void)initUI
{    
    UIScrollView *view = [[UIScrollView alloc] initWithFrame:self.bounds];
    view.backgroundColor = [UIColor magentaColor];
    view.contentSize = CGSizeMake(self.bounds.size.width, self.bounds.size.height * 10);
    [self addSubview:view];
}

下面看输出结果

2017-12-10 11:12:48.550635+0800 JJLayoutSubviews[2007:71395] initWithFrame
2017-12-10 11:14:01.519347+0800 JJLayoutSubviews[2007:71395] layoutSubviews被调用了
2017-12-10 11:14:01.520385+0800 JJLayoutSubviews[2007:71395] layoutSubviews被调用了

从这里可以看见,滚动子视图的scrollView是不会调用其所在的父视图的layoutSubviews方法的。

6. 屏幕旋转

下面我们就看一下旋转屏幕的情况,运行在真机并进行横竖屏切换并查看输出结果。

下面看输出结果

2017-12-10 13:16:27.955418+0800 JJLayoutSubviews[14428:1263114] initWithFrame
2017-12-10 13:16:27.959850+0800 JJLayoutSubviews[14428:1263114] layoutSubviews被调用了
2017-12-10 13:16:27.959937+0800 JJLayoutSubviews[14428:1263114] layoutSubviews被调用了
2017-12-10 13:16:30.511200+0800 JJLayoutSubviews[14428:1263114] layoutSubviews被调用了

最后面,2017-12-10 13:16:30.511200+0800 JJLayoutSubviews[14428:1263114] layoutSubviews被调用了就是旋转屏幕调用的,也就是说旋转屏幕会调用视图layoutSubviews方法。

7. setNeedsLayout

我们知道这个方法,setNeedsLayout,这个可以对视图进行标记,标记为需要布局更新,但是不会立即更新,而是会等待下一个运行循环才会去更新。

  • setNeedsLayout方法: 标记为需要重新布局,异步调用layoutIfNeeded刷新布局,不立即刷新,但layoutSubviews一定会被调用

  • layoutIfNeeded方法:如果,有需要刷新的标记,立即调用layoutSubviews进行布局(如果没有标记,不会调用layoutSubviews)

如果要立即刷新,要先调用[view setNeedsLayout],把标记设为需要布局,然后马上调用[view layoutIfNeeded],实现布局

在视图第一次显示之前,标记总是“需要刷新”的,可以直接调用[view layoutIfNeeded]

下面看一下代码,在点击事件中添加代码。

- (void)viewDidTapped
{
    [self setNeedsLayout];
    [self layoutIfNeeded];
}

下面看输出结果

2017-12-10 13:27:57.545089+0800 JJLayoutSubviews[14440:1266773] initWithFrame
2017-12-10 13:27:57.550090+0800 JJLayoutSubviews[14440:1266773] layoutSubviews被调用了
2017-12-10 13:28:01.546769+0800 JJLayoutSubviews[14440:1266773] layoutSubviews被调用了
2017-12-10 13:28:02.362688+0800 JJLayoutSubviews[14440:1266773] layoutSubviews被调用了
2017-12-10 13:28:02.897325+0800 JJLayoutSubviews[14440:1266773] layoutSubviews被调用了

后面几条就是点击屏幕打印出来的,说明,setNeedsLayout对视图进行标记并layoutIfNeeded执行视图layoutSubviews方法。

后记

未完,待续~~~

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

推荐阅读更多精彩内容