View controller是iOS中顶层的视图载体和控制器,它需要对view负责,管理view的生命周期,相关处室话以及交互事件,除此以外还需要为view提供合适的数据,以供view使用。
View controller与view之间的绑定是十分密切的,它也一样参与事件的响应,并管理事件。对于这点,我们可以观察UIViewController的继承结构:
NSObject->UIResponder->UIViewController
从中我们可以看出view controller是一个UIResponder对象,它在事件响应链中,处在view controller的root view以及view的superview之间。如果view controller中的views没有控制某个事件,那么view controller会将该事件沿着响应链传递给superview.
每个view controller都会管理view的层次结构。View controller的root view则保存在view属性中,而root view相当于view层次结构中的容器。
对于root view而言,它的size和position是由它的父级结构决定,比如说某个parent view controller,container view controller或者app的window。
管理视图
view的创建
在view controller中,有以下几种方法创建root view:
通过storyboard文件来创建。
通过nib文件来创建。
通过loadView方法来创建。
在view controller构建view的过程中,view controller首先检查是否有指定的nib文件(其中storyboard也会提供nib文件)。 如果有,那么view controller通过nib文件进行创建。如果没有,那么view controller会检查loadView是否有创建。如果loadView也没有那么view controller则会自动生成一个默认view来使用。
不过这里需要注意,当已经通过nib文件或者storyboard创建view的时候,请不要再重载loadView方法。
如果非要重载,可以这么实现,但是不推荐:
- (void)loadView { [super loadView]; <#code#> }
view与惰性加载
在view controller中,view的创建采用的是惰性加载的方式。即当view被引用的时候,view才会被创建。比如说将view添加到某个视图层次结构中的时候,再比如以下的情况:
UIViewController *vc = [UIViewController new]; vc.view.backgroundColor = [UIColor redColor];
像这样的直接引用view,也可以触发view的创建。
除此以外还可以通过loadViewIfNeeded主动触发view的创建。
如果我只想判断当前的view是否存在而不需要触发view的创建,那改怎么做呢? 可以通过isViewLoaded进行判断。
直接给view赋值
那么除了常规的创建以外,view controller的view可以直接被赋值么? 答案是可以的。
UIViewController *vc = [UIViewController new]; UIView *myView = [UIView new]; myView.backgroundColor = [UIColor yellowColor]; vc.view = myView;
awakeFromNib方法
当view controller通过nib文件加载后,那么就会调用这个方法。
这个方法可以保证,outlets和actions已经被正确的连接。 但是并没有对于nib中创建对象的顺序作出保证,因此请不要在awakeFromNib时间点之前使用这些对象(比如说初始化方法)。
你可以通过重载awakeFromNib方法,来完成对对象的一些额外的配置,但要注意,请一定要在awakeFromNib方法中调用其父类的方法。
重载loadView
loadView的主要职责在于创建view controller的root view。它可以通过nib文件以及代码进行创建。如果你需要用代码自定义创建一个root view,那么需要重载这个方法,不过你要确保,你所创建的这个view有且仅有当前view controller才能持有。 不过要注意,在重载的时候,不能够调用[super loadView]。
1 - (void)loadView { 2 UIView *view = [UIView new]; 3 view.frame = [UIScreen mainScreen].bounds; 4 view.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleHeight; 5 self.view = view; 6 }
viewDidLoad方法
不论view是通过nib文件创建还是通过loadView自定义创建,在viewDidLoad函数中,它都可以确保view controller的view是一定存在的。 所以可以在这个方法做一些关于view的额外的配置。比如说添加额外的subview等等。
view的释放
任何时候你都应该确保,view controller是view的唯一管理者,如果当前的view被其它的对象引用了,那么view controller的管理便则有可能被限制。
在view controller的生命周期中,view是可以被释放的。当然这样的做法并不普遍,一般会在内存紧张的时候,进行对view的释放。
1 - (void)didReceiveMemoryWarning { 2 [super didReceiveMemoryWarning]; 3 if ([self isViewLoaded] && self.view.window == nil) { 4 self.view = nil; 5 NSLog(@"view ==> %@", [self viewIfLoaded]); 6 } 7 }
响应视图相关的通知
当views的显示状态改变了的时候,view controller会自动给予通知。
viewWillAppear:
在view被添加到视图层次结构之前调用。
可以在这期间为view准备数据,调整status bar等。
viewDidAppear:
在view被添加到视图层次结构之后调用。
viewWillDisapper:
在view被移除视图层次结构之前调用。
这时候可以做一些清理工作,比如一些可以易于重新生成的数据。
viewDidDisappear:
在view移除视图层次结构之后调用。