生命周期
我们建立一个简单的模型来测试生命周期,新建两个viewController,一个是主视图控制器(mainViewController,以下简称mainVC),一个是副视图控制器(sub ViewController ,以下简称subVC),在mainVC里点击一个button,以modal方式切换至subVC,然后在subVC里点击另一个button关闭subVC。我们将这两个控制器的每个状态都打印出来,各个阶段的执行如下:
case1.第一次运行app:
main loadView
main viewDidLoad
main viewWillAppear
main viewDidAppear
case 2.在mainVC里点击Button,以modal方式切换至subVC:
sub loadView
sub viewDidLoad
main viewWillDisappear
sub viewWillAppear
sub viewDidAppear
main viewDidDisappear
case 3.在subVC里点击button关闭subVC并返回mainVC:
sub viewWillDisappear
main viewWillAppear
main viewDidAppear
sub viewDidDisappear
sub dealloc
当一个视图控制器被创建,并在屏幕上显示的时候代码的执行顺序:
step1 alloc对象,分配空间
step2,init (initWithNibName)初始化对象
step3,loadView 从nib载入视图,通常这一步不需要去干涉,除非你没用使用xib文件创建视图
step4,viewDidLoad载入完成,可以进行自定义数据以及动态创建其他控件
step5,viewWillAppear视图将出现在屏幕之前,马上这个视图就会被展现在屏幕上了
step6,viewDidAppear视图已经在屏幕上渲染完成
当一个视图控制器被移除屏幕并且销毁的时候的执行顺序:
step1 viewWillDisappear视图将从屏幕上移除之前执行
step2 viewDidDisappear视图已经被从屏幕上移除,用户看不到这个视图了
step3 dealloc视图被销毁
loadView和viewDidLoad的区别:当loadView时,还没有view,而viewDidLoad时,view已经创建好了。详细的加载循环:
step1 程序请求viewController的view属性
step2 如果view在内存中,则直接加载,如果不存在,则调用loadView方法
step3 loadView方法执行如下方法:
* 如果重载了这个方法,则必须创建必要的UIView并且将一个非nil值传给ViewController的view属性
* 如果没用重载这个方法,viewController会默认使用自己的nibName和nibBundle属性尝试从nib文件加载view。如果没用找到nib文件,它尝试寻找一个与viewController类名匹配的nib文件
* 如果没有可用的nib文件,那么它创建一个空的UIView作为它的view
最后还要考虑一个重要的情况:内存不足警告。当程序员收到内存警告的时候,会调用每一个ViewController的didReceiveMemoryWarning方法,我们需要作出响应,释放程序中暂时不需要的资源,通常都会重写该方法,但记得重写的时候要调用super的该方法。
iOS 3.0-iOS6.0 期间,didReceiveMemoryWarning方法会判断当前viewController的viewDidUnload方法。但是iOS6.0开始,ViewDidUnload和viewWillUnload这两个方法已被废除,收到lowmemory时系统不会释放view,而只是释放controller的resource。
一种常见处理内存警告的方式:
- (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
float ver = [[UIDevice currentDevice] systemVersin] floatValue];
if(ver >= 6.0f)
{
if(self.isViewLoaded && !self.view.window)
{
self.view = nil; // 确保下次重新加载
}
}
}
上面的代码先取得当前iOS系统的版本号,如果是iOS6.0或以上版本,进一步判断视图是否会被装载进内存,并且是否为当前视图,在这两个条件都满足(已经装载进内存&&不是当前视图)时,将self.view设置为nil,这样就能保证再调用该viewController时,loadView和viewDidLoad被再次调用。
我们在xcode调试器里模拟内存警告,监控此时切换的状态:
case 4, 当已切换至subVC,模拟内存警告,并返回mainVC,不处理didReceiveMomeoryWarning
Received memory warning.
main didReceiveMemoryWarning
sub viewWillDisappear
main viewWillAppear
main viewDidAppear
sub viewDidDisappear
sub dealloc
case 5.当切换至subVC,模拟内存警告,并返回mainVC,处理didReceiveMemoryWarning。
main didReceiveMemoryWarning
sub didReceiveMemoryWarning
main loadView
main viewDidLoad
sub viewWillDisappear
main viewWillAppear
main viewDidAppear
sub viewDidDisappear
sub dealloc
可以看出,当处理了didReceivememoryWarning后,重新执行了非当前视图的loadView和viewDidLoad方法。