说来惭愧,苹果在ios5就提供了addChildViewController这一方法,而我却一直没有使用过,也对它不甚了解,这次有机会用用,仔细学习下。
方法定义
swift中,对应的方法是:
/*
If the child controller has a different parent controller, it will first be removed from its current parent
by calling removeFromParentViewController. If this method is overridden then the super implementation must
be called.
大意是:如果被添加的VC已经有父VC了,在调用这个方法之前会先从父VC中移除,然后再被添加到目前的VC上,做为目前VC的子VC。
*/
@available(iOS 5.0, *)
open func addChild(_ childController: UIViewController)
苹果对这一方法的解释:
This method creates a parent-child relationship between the current view controller and the object in the childController parameter. This relationship is necessary when embedding the child view controller’s view into the current view controller’s content. If the new child view controller is already the child of a container view controller, it is removed from that container before being added.
This method is only intended to be called by an implementation of a custom container view controller. If you override this method, you must call super in your implementation.
大意是:
这一方法在 当前的VC 和 被添加的VC 之间创建了一种父-子的关系。当把一个VC的view添加到当前VC的view上的时候,就必须先使用addChild这个方法添加这种父子关系。如果子VC已经有父VC了,那么调用该方法会先把子VC从之前的父VC中移除,然后再添加到当前的VC上作为子VC。
这个方法只会在自定义的VC中被调用,如果打算重写这个方法,则必须调用super的此方法。
也就是说,苹果iOS5之后的API增加的 addChild(_ childController: UIViewController) 方法,并且希望我们在使用addSubview时,同时调用addChild(vc)方法将sub view对应的viewController也加到当前ViewController的管理中。
对于那些当前暂时不需要显示的subview,只通过addChild(_ childController: UIViewController)把subViewController加进去;需要显示时再调用transitionFromViewController方法。将其添加进入底层的ViewController中。
这样做的好处:
1.无疑,对页面中的逻辑更加分明了。相应的View对应相应的ViewController。
2.当某个子View没有显示时,将不会被Load,减少了内存的使用。
3.当内存紧张时,没有Load的View将被首先释放,优化了程序的内存释放机制。
- 在iOS5中,ViewController中新添加了下面几个方法:
- addChildViewController:
- removeFromParentViewController
- transitionFromViewController:toViewController:duration:options:animations:completion:
- willMoveToParentViewController:
- didMoveToParentViewController:
使用场景:
我们的日常开发中常有这样一种需求,通过切换标签来切换不同的页面,如果在一个 controller 管理这些 view 的话,代码就显得耦合度过高,也与 Apple 的 MVC 模式不符合,Apple 推荐一个 controller 管理一个 view。 同样有人建议类似 [self.view addSubview:viewController.view] 的方式用另一个 controller 来控制,但是这样会产生一个新的问题:直接 add 进去的subView不在 viewController的 view hierarchy 内,事件不会正常传递,如:旋转、触摸等,属于危险操作违背CocoaTouch开发的设计MVC原则,viewController应该且只应该管理一个view hierarchy。同样我们也要考虑另一个问题:这些子 view 大多数不会一直处于界面上,只是在某些情况下才会出现,例如登陆失败的提示 view,上传附件成功的提示 view,网络失败的提示 view 等。但是虽然这些 view 很少出现,但是我们却常常一直把它们放在内存中。
关于子VC的生命周期:
当childViewController没有被加到任何父视图控制器时,如果把childViewController的view加到别的视图上,viewWillAppear和viewDidAppear会正常调用。但是当childViewController被加到一个父视图控制器上后,viewWillAppear和viewDidAppear就会与父视图控制器的viewWillAppear和viewDidAppear事件同步。
所以调用先后问题要注意:
先调用addSubView,viewWillAppear和viewDidAppear会各调用一次,再addChildViewController,与父视图控制器的事件同步,即当父视图控制器的viewDidAppear调用时,childViewController的viewDidAppear方法会再调用一次。所以viewDidAppear方法被调用了两次。
先调用addChildViewController,childViewController的事件与父视图控制器同步,当父视图控制器的viewDidAppear调用时,childViewController的viewDidAppear方法会调用一次,再调用addSubView也不会触发viewWillAppear和viewDidAppear。