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

当我们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的再次执行了。

你可能感兴趣的:(viewDidLoad和构造函数你真的可能忽略的点)