当前位置: 首页 > article >正文

「iOS」viewController的生命周期

iOS学习

  • ViewController生命周期有关方法
  • 案例
  • 注意

ViewController生命周期有关方法

  1. init - 初始化程序;
  2. loadView - 在UIViewController对象的view被访问且为空的时候调用;
  3. viewDidLoad - 视图加载完成后调用;
  4. viewWillAppear - UIViewController对象的视图即将加入窗口时调用;
  5. viewDidAppear - UIViewController对象的视图已经加入窗口时调用;
  6. viewWillDisappear - UIViewController对象的视图即将消失时调用;
  7. viewDidDisappear - UIViewController对象的视图已经消失时调用;
  8. didReceiveMemoryWarning - 出现内存警告。由于此处ViewController的view已经被释放了,ViewController中的空间可以在此处释放掉。
  9. dealloc - 视图被销毁,此次需要对在init和viewDidLoad中创建的对象进行释放;

需要注意的是:
loadView和viewDidLoad的区别就是,loadView时view还没有生成,viewDidLoad时,view已经生成了,loadView只会被调用一次,而viewDidLoad可能会被调用多次(View可能会被多次加载),当view被添加到其他view中之前,会调用viewWillAppear,之后会调用viewDidAppear。
当view从其他view中移除之前,调用viewWillDisAppear,移除之后会调用viewDidDisappear。
当view不再使用时,受到内存警告时,ViewController会将view释放并将其指向为nil。


案例

创建A,B两个视图。
其中A视图代码:


#import "ViewController.h"
#import "secondVC.h"

@interface ViewController ()

@end

@implementation ViewController



- (void)viewDidLoad {
    [super viewDidLoad];
    NSLog(@"view1 Did Load");

    // Do any additional setup after loading the view.
    self.btn = [UIButton buttonWithType:UIButtonTypeSystem];
    [self.btn setTitle:@"切换视图" forState:UIControlStateNormal];
    self.btn.frame = CGRectMake(100, 100, 100, 50);
    self.btn.center = self.view.center;
    [self.btn addTarget:self action:@selector(pressbtn) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:self.btn];
    
}

-(void)pressbtn
{
    secondVC *secondvc = [[secondVC alloc] init];
    [self presentViewController:secondvc animated:YES completion:nil];
}
-(void)viewWillAppear:(BOOL)animated
{
    NSLog(@"view1将要显示");
}

-(void)viewDidAppear:(BOOL)animated
{
    NSLog(@"view1已经显示");
}

-(void)viewWillDisappear:(BOOL)animated
{
    NSLog(@"view1将要消失");
}

-(void)viewDidDisappear:(BOOL)animated
{
    NSLog(@"view1已经消失");
}
@end

B视图代码:


#import "secondVC.h"

@interface secondVC ()

@end

@implementation secondVC

-(void)loadView
{
    self.view = [[UIView alloc] init];
    NSLog(@"view2正在加载");
}

- (void)viewDidLoad {
    [super viewDidLoad];
    NSLog(@"view2 Did Load");

    // Do any additional setup after loading the view.
    self.view.backgroundColor = [UIColor blueColor];
    UIView *view = [[UIView alloc] init];
    view.backgroundColor = [UIColor redColor];
    view.frame = CGRectMake(150, 300, 100, 100);
    [self.view addSubview:view];
    
    UIButton *btn = [UIButton buttonWithType:UIButtonTypeSystem];
    [btn setTitle:@"切换视图" forState:UIControlStateNormal];
    btn.frame = CGRectMake(100, 700, 100, 80);
    [btn addTarget:self action:@selector(press) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:btn];
    
}

-(void)viewWillAppear:(BOOL)animated
{
    NSLog(@"view2将要显示");
}

-(void)viewDidAppear:(BOOL)animated
{
    NSLog(@"view2已经显示");
}

-(void)viewWillDisappear:(BOOL)animated
{
    NSLog(@"view2将要消失");
}

-(void)viewDidDisappear:(BOOL)animated
{
    NSLog(@"view2已经消失");
}

-(void)press
{
    [self dismissViewControllerAnimated:YES completion:nil];
}

@end

最后我们得到运行结果:
第一个视图显示:
请添加图片描述
切换第二个视图:

请添加图片描述
返回第一个视图:
请添加图片描述
使用模态切换的方式切换视图时,会将第一个视图重新下载一遍。即重新执行一遍
viewDidLoad 方法。
而使用push方法切换视图时,则会发现,在第一个视图完全消失后,第二个视图才会加载好。这里直接给出运行结果。
请添加图片描述


注意

如果需要使用loadView方法,要注意对view进行初始化,否则会进入死循环。

原因:若 loadView 没有加载 view,即为 nil,viewDidLoad 会一直调用 loadView 加载 view,因此构成了死循环,程序即卡死。

解决方法:

  1. 对view进行初始化赋值
  2. 使用[Supper LoadView]方法
  3. 在 ViewDidLoad 里创建 view,省略loadView方法。

http://www.kler.cn/a/314688.html

相关文章:

  • Babylon.js行为编写及使用参考
  • 【C++】C++11(二)
  • VSCode 在Windows下开发时使用Cmake Tools时输出Log乱码以及CPP文件乱码的终极解决方案
  • 在iStoreOS上安装Tailscale
  • vivado时序约束和优化
  • 【mysql】流程控制
  • Android TV RecyclerView列表获得焦点左右换行
  • 如何在Mac上安装多个Python环境
  • Spring Mybatis PageHelper分页插件 总结
  • MySQL篇(SQL优化)(持续更新迭代)
  • Android Studio 开发快速获取开发版和发布版SHA1和MD5
  • 汽车美容服务管理系统的数据库设计与数据操作
  • nvm 下载node报错:Could not retrieve https://nodejs.org/dist/index.json.
  • 奇安信渗透2面经验分享
  • HarmonyOS第一课-应用程序框架基础习题答案
  • 邮件安全治理
  • 解决nginx代理SSE接口的响应没有流式返回
  • 详细分析Spring的动态代理机制
  • zynq的PS端mac与RTL8211F的连接要点
  • 微服务架构中的负载均衡与服务注册中心(Nacos)
  • Cursor免费 GPT-4 IDE 工具的保姆级使用教程
  • Spring01
  • C# 中的NPOI 库
  • 生物信息常用编辑器:轻量高效的VS Code
  • 山东潍坊戴尔存储服务器维修 md3800f raid恢复
  • Docker Registry API best practice 【Docker Registry API 最佳实践】