当我们push一个ViewController的时候,这个viewController的viewDidLoad方法什么时候开始执行的呢?我猜一部分同学可能并不是很清楚,但是他们肯定知道,push viewController的时候,这个viewController的viewDidLoad方法肯定会执行。这篇文章我会对这个问题说说自己的理解,以便新同学学习和给自己凑数一篇笔记。
1. 为什么需要清楚的知道这个问题
通过示例代码,举个场景说明此问题
class ControllerB: UIViewController {
var isShowBottom: Bool
init(isShowBottom: Bool = true) {
self.isShowBottom = isShowBottom
super.init()
}
override func viewDidLoad() {
super.viewDidLoad()
if isShowBottom {
view.addSubview(bottomView)
}
}
}
// 方式1
let vc = ControllerB(isShowBottom: false)
navigationController?.pushViewController(vc, animated: true)
// 方式2
let vc = ControllerB()
vc.isShowBottom = false
navigationController?.pushViewController(vc, animated: true)
如果viewDidLoad是在调用构造函数的时候执行的,那么方式2
就会有问题,因为我们是想isShowBottom为false的,而执行ControllerB()的时候,内部把viewDidLoad也执行了,即在执行vc.isShowBottom = false
的时候,bottomView就已经被add到ContrllerB的view上了。
如果viewDidLoad是在执行navigationController?.pushViewController(vc, animated: true)
这句代码的时候执行的,那么方式1
和方式2
效果一样。
通过这个例子,你应该领悟到我们为什么应该清楚的知道viewDidLoad调用时机了吧。
2. 通过一个例子引出结论
class BaseController: UIViewController {
override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?) {
super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil)
commonInit()
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
private func commonInit() {
hidesBottomBarWhenPushed = true
// view.backgroundColor = UIColor.red #注释1
print("test")
}
override func viewDidLoad() {
super.viewDidLoad()
// view.backgroundColor = UIColor.red #注释2
}
}
class ControllerB: BaseController {
override func viewDidLoad() {
super.viewDidLoad()
view.backgroundColor = UIColor.blue
}
}
let vc = ControllerB()
navigationController?.pushViewController(vc, animated: true)
controllerB
继承自BaseController
,controllerB在viewDidLoad中设置了view.backgroundColor = UIColor.blue
,那么controllerB就真的是蓝色了吗?
答案当然是NO。
打开注释1
,关闭注释2
: controllerB.view显示出来的就是红色;
打开注释2
,关闭注释1
: controllerB.view显示出来的就是蓝色;
这个问题的关键点就是,当执行commonInit()
方法中的view.backgroundColor = UIColor.red
的时候,会触发viewDidLoad()
的执行,执行完viewDidLoad方法后,然后又把red设置给controller的view了。所以无论你在子类的viewDidLoad()中如何设置view.backgroundColor,最终都会被red给覆盖。如果打开注释2
,关闭注释1
,这种情况就正常了,在父类中设置红色,然后在子类中设置蓝色,最终显示蓝色。
结论
每次访问UIViewController的view而且view为nil,loadView方法就会被调用。系统是在loadView方法中创建好UIViewController的view的。
viewDidLoad是在loadView方法创建view完毕后被调用的。
打开注释1
,关闭注释2
,当在commonInit()
中,首次访问controller的view,这个时候会调用系统的loadView()方法,在loadView()方法中创建完view后就会开始执行viewDidLoad方法,执行完viewDidLoad后,赋值backgroundColor红色,之后再执行print("test")。所以显示红色也就不奇怪了,这个过程想测试的可以通过断点调试。
通过结论也顺带可以理解出,当我们从controllerA push controllerB, 然后pop掉controllerB,controllerA的viewDidLoad并不会再次执行,因为controllerA的view已经有值了,并不会触发loadView()的执行,从而也就不会触发viewDidLoad的再次执行了。