viewDidLoad和构造函数你真的可能忽略的点

当我们push一个ViewController的时候,这个viewController的viewDidLoad方法什么时候开始执行的呢?我猜一部分同学可能并不是很清楚,但是他们肯定知道,push viewController的时候,这个viewController的viewDidLoad方法肯定会执行。这篇文章我会对这个问题说说自己的理解,以便新同学学习和给自己凑数一篇笔记。
1. 为什么需要清楚的知道这个问题
【viewDidLoad和构造函数你真的可能忽略的点】通过示例代码,举个场景说明此问题

class ControllerB: UIViewController { var isShowBottom: Boolinit(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的再次执行了。

    推荐阅读