结论
使用 UINavigationController pushViewController:animated: API 显示的 UIViewController,它的整个生命周期其实和 API 中传递的 animated 参数值有关
过程
让我们来做个小 DEMO 实验来验证这个结论
首先我们的 DEMO 很简单,只有两个页面,ViewController 和 AnotherViewController
然后我们使用默认的 StoryBoard 的方式来构建我们的 App UI 界面,App 主 Window 的 RootViewController 是一个 UINavigationController,UINavigationController 的 RootViewController 是 ViewController
ViewController 上面有两个按钮,两个按钮都绑定了跳转到 AnotherViewController 的 Segue,只是一个有动画,一个没有动画
接着我们在 AnotherViewController 里添加 UIViewController 整个生命周期内的各个函数代码 Log 打点
运行 App
点击“显示页面 B(有动画)”按钮后 Xcode 控制台上打印如下:
和平常一样,很正常,没什么好说的
但是当我们再点击“显示页面 B(没有动画)”按钮后 Xcode 控制台上打印如下:
很明显看到 loadViewIfNeeded 由之前的在 viewWillAppear: 之后调用提前到了 loadView 之前,并且 viewDidAppear: 和 viewWillLayoutSubviews、viewDidLayoutSubviews 之间的调用顺序也改变了,viewDidAppear: 函数调用比 viewWillLayoutSubviews、viewDidLayoutSubviews 还早
彩蛋
在这过程中意外地发现,如果是没有动画显示 AnotherViewController 的时候,断点在 viewDidAppear:,可以看到其实这时候 AnotherViewController 根本还没有显示
结语
所以 viewDidAppear: 在整个 UIViewController 的生命周期中是否准确,取决于 animated 这个参数值