1、前言

在面试中,面试官可能会问这样的问题,loadView有什么作用,它与viewDidLoad有何区别

首先我们得知道,控制器view是通过懒加载的方式进行加载的,即用到的时候再加载。




2、loadView方法

当我们用到控制器view时,就会调用控制器view的get方法,在get方法内部,首先判断view是否已经创建,如果已存在,则直接返回存在的view,如果不存在,则调用控制器的loadView方法,在控制器没有被销毁的情况下,loadView也可能会被执行多次。




3、viewDidLoad方法

当控制器的loadView方法执行完毕,view被创建成功后,就会执行viewDidLoad方法,该方法与loadView方法一样,也有可能被执行多次。在开发中,我们可能从未遇到过执行多次的情况,那什么时候会执行多次呢?


比如A控制器push出B控制器,此时,窗口显示的是B控制器的view,此时如果收到内存警告,我们一般会将A控制器中没用的变量及view销毁掉,之后当我们从B控制器pop到A控制器时,就会再次执行A控制器的loadView方法与viewDidLoad方法。

如下图所示,注意控制台打印:

iOS之深入了解控制器View的加载 1

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 1





4、控制器view的加载

先看一下Demo的文件结构,ViewController为A控制器,TestViewController为B控制器:

iOS之深入了解控制器View的加载 2

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 2




4.1、通过storyboard加载

当控制器通过storyboard加载时,需要指定storyboard的名称,控制器view最终就是storyboard所描述的样子,这个比较简单,不做详细阐述。

[Objective-C] 纯文本查看 复制代码
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
  UIStoryboard *storyboard = [UIStoryboard storyboardWithName:@"TestViewController" bundle:nil];
  TestViewController *testVC = [storyboard instantiateInitialViewController];
  [self.navigationController pushViewController:testVC animated:YES];
}




4.2、通过xib加载

当控制器view通过xib加载的时候,可能会出现三种情况。


4.2.1、指定xib名称(OtherViewController.xib)

[Objective-C] 纯文本查看 复制代码
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
  TestViewController *testVC = [[TestViewController alloc] initWithNibName:@"OtherViewController" bundle:nil];
  [self.navigationController pushViewController:testVC animated:YES];
}


当我们指定了xib的名称,loadView方法就会去加载对应的xib(OtherViewController.xib),最终是这个样子的

iOS之深入了解控制器View的加载 3

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 3



4.2.2、不指定xib名称1

[Objective-C] 纯文本查看 复制代码
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
  TestViewController *testVC = [[TestViewController alloc] init];
  [self.navigationController pushViewController:testVC animated:YES];
}


如果我们不指定xib名称,loadView就会加载与控制器同名的xib(TestViewController.xib),最终是这个样子的

iOS之深入了解控制器View的加载 4

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 4



4.2.3、不指定xib名称2

我们先将TestViewController.xib这个文件删除掉,这个时候,我们再来运行程序,结果是这样的

iOS之深入了解控制器View的加载 5

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 5


根据上图我们可以得知,当没有指定xib名称,且没有与控制器同名的xib时,会加载前缀与控制器名相同而不带controller的xib(TestView.xib)。



4.3、不通过sb\xib加载

将TestView.xib这个文件也删除掉,再来运行程序,结果是这样的

iOS之深入了解控制器View的加载 6

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 6


这么黑,难道没有创建控制器view?

iOS之深入了解控制器View的加载 7

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 7


如上图,控制器view是存在的,只不过颜色为clearColor,所以看到的黑色其实是UIWindow的



4.4、重写loadView方法

我们重写TestViewController的loadView方法,里面不做任何事

[Objective-C] 纯文本查看 复制代码
- (void)loadView {
}


运行程序看结果

iOS之深入了解控制器View的加载 8

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 8


结果跟上面一样黑,不同的是,这次并没有创建view,注意看上图最外层并不是UIView

如果我们希望控制器view加载出来的时候不是UIView而是其他控件,比如UIImageView,那我们就可以重写loadView

[Objective-C] 纯文本查看 复制代码
- (void)loadView{
  self.view = [[UIImageView alloc] init];
}


iOS之深入了解控制器View的加载 9

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 9





5、结论

1.重写loadView方法,则会根据重写的loadView方法创建view

2.控制器通过storyboard加载,则根据storyboard的描述创建view

3.控制器view通过xib加载,则根据nibName对应的xib创建view

4.没有指定nibName,则根据与控制器同名的xib创建view

5.没有同名的xib,则根据与控制器名前缀相同不带controller的xib创建view

6.如果都没有,则创建一个空白的xib




6、小细节

在上面的2、3两点结论中,不知道大家有没有一个疑问

iOS之深入了解控制器View的加载 10

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 10


为什么上面是说的控制器,而下面却说的控制器view?

笔者结合控制器的awakeFromNib方法给大家说明一下这个问题

顾名思义,当控制器从nib加载的时候就会调用这个方法


先来看看通过storyboard加载的情况:

[Objective-C] 纯文本查看 复制代码
//A控制器中代码
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
  UIStoryboard *storyboard = [UIStoryboard storyboardWithName:@"TestViewController" bundle:nil];
  TestViewController *testVC = [storyboard instantiateInitialViewController];
  [self.navigationController pushViewController:testVC animated:YES];
}
//B控制器中代码
- (void)awakeFromNib {
  NSLog(@"B通过nib加载");
}


iOS之深入了解控制器View的加载 11

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 11


控制台打印了“B通过nib加载”,即调用了B控制器的awakeFromNib方法。

将之前删除的TestViewController.xib文件重写添加进去,再来看通过xib加载的情况:

[Objective-C] 纯文本查看 复制代码
//A控制器中代码改为如下
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
  TestViewController *testVC =[[TestViewController alloc] init];
  [self.navigationController pushViewController:testVC animated:YES];
}
//B控制器中代码不变


iOS之深入了解控制器View的加载 12

iOS之深入了解控制器View的加载 - 敏捷大拇指 - iOS之深入了解控制器View的加载 12


控制台没有任何输出,即B控制器的awakeFromNib方法并没有被调用



6.1、结论

storyboard加载的是控制器及控制器view,而xib加载的仅仅只是控制器的view




作者:codingZero