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的再次执行了。
推荐阅读
- 急于表达——往往欲速则不达
- 第三节|第三节 快乐和幸福(12)
- 20170612时间和注意力开销记录
- 2.6|2.6 Photoshop操作步骤的撤消和重做 [Ps教程]
- 对称加密和非对称加密的区别
- 眼光要放高远
- 樱花雨
- 前任
- 2020-04-07vue中Axios的封装和API接口的管理
- 烦恼和幸福