从loadView()和viewDidLoad()的区别说起

引言

loadView()和viewDidLoad()是入行iOS开发经常遇到的两个方法。记得当初也看了Apple官网提供的手册和资料,然而实际开发起来,仍不知怎么区分。今天就结合UIViewController的生命周期好好理理loadView()和viewDidLoad()的区别。

加载顺序

class ViewController: UIViewController {
 
override func loadView() {
   print("loadView")
 }

 override func viewDidLoad() {
    print("viewDidLoad")
    super.viewDidLoad()
 }

 override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
 }
}

直接执行后,输出结果:

从loadView()和viewDidLoad()的区别说起_第1张图片

可以看到先执行了loadView(),然后才执行viewDidLoad()。

UIViewController的生命周期

从loadView()和viewDidLoad()的区别说起_第2张图片
UIViewController生命周期

套用网上的一张UIViewController中view的生命周期截图。

  • 当调用UIViewController中view属性的getter方法时,如果此时view 为nil,则系统总是会自动调用loadView()方法(Apple的文档也说了,该方法永远不能手动调用);
  • 如果loadView()方法没有被override,则默认会生成一个白色背景、大小与屏幕一致的视图赋值给该view属性;如下图所示:
  class ViewController: UIViewController {
 
 // override func loadView() {
  //  print("loadView")
// }

  override func viewDidLoad() {
      print("viewDidLoad")
      super.viewDidLoad()
 
 }
  override func didReceiveMemoryWarning() {
      super.didReceiveMemoryWarning()
      // Dispose of any resources that can be recreated.
   }
 }
从loadView()和viewDidLoad()的区别说起_第3张图片
  • 如果该方法被override,则必须在方法中为self.view赋值,否则没有view被加入到view hierarchy中,因此只显示黑色的Window背景色;
    使用本文文首处的代码,结果如下图所示:


    从loadView()和viewDidLoad()的区别说起_第4张图片

    在override的loadView()中为view赋值:

override func loadView() {
    print("loadView")
    let view = UIView(frame: UIScreen.mainScreen().bounds)
    view.backgroundColor = UIColor.blueColor()
    self.view = view
 }

结果如下图所示:


从loadView()和viewDidLoad()的区别说起_第5张图片
  • 每当loadView()调用完成后,系统会自动调用viewDidLoad()方法,这与loadView()是否被override无关。

控制台中的两遍结果

那么,为什么文章起始处的控制台输出了两遍"loadView"和"viewDidLoad"?

可以大胆猜想这应该是系统访问了两次viewController的view属性。由于view一直没有被赋值,因此每次都会调用loadView()和viewDidLoad()。这也说明了,如果不override loadView()或在override的loadView()为self.view正确赋值后,控制台将会只输出一次“viewDidLoad"。

总结

  • 一般不必override loadView()方法;若override,则必须负责为view赋值;
  • 一般override viewDidLoad()方法; 此时self.view已经有值了,可以直接使用addSubView添加子view到其中;
  • 一般override viewWillLayoutSubviews()方法;在此方法中调整各子view的frame。

你可能感兴趣的:(从loadView()和viewDidLoad()的区别说起)