viewDidLoad和awakeFromNib时序

时间:2011-06-10 06:14:37

标签: iphone ipad uitableview viewdidload awakefromnib

据我了解,将始终在viewDidLoad之前调用awakeFromNib。

所以我有一个UITableViewController的子类,它是从xib文件中取消归档的。

我在里面定义了这两个方法:

- (void)awakeFromNib {
  [super awakeFromNib];
  NSLog(@"awake from nib");
}

- (void)viewDidLoad {
  [super viewDidLoad];
  NSLog(@"view did load");
}

在控制台中“从nib唤醒”之前会出现“view does load”。我试图在[超级awakeFromNib]上使用断点,并反复点击F7(Step Into),令我惊讶的是,它进入 - (void)viewDidLoad,然后转到awakeFromNib内的第二行。

有没有人知道这里发生了什么?我在常规UIViewController的子类中做了完全相同的事情,并且日志语句是相反的,正如我最初预期的那样......

4 个答案:

答案 0 :(得分:12)

要了解这一事实,我建议您查看NSBundle的{​​{3}}方法。

从nib初始化视图控制器时,首先它会加载其中包含的视图,然后根据nib设置文件所有者属性。将文件所有者的viewDidLoad属性设置为其中一个已加载的视图时,将调用view方法。并且在设置所有文件所有者出口和属性(包括awakeFromNib属性)时调用view。因此,有必要在viewDidLoad之前调用awakeFromNib

希望这会有所帮助

答案 1 :(得分:5)

我认为你不必在超类上调用awakeFromNib。

检查this

修改

我刚刚进行了快速测试,结果如下:

场景1 : MainWindow.Xib有一个UIViewController子类TestingAwakeFromNibViewController,它有自己的Nib文件TestingAwakeFromNibViewController.xib

TestingAwakeFromNibViewController有一个名为btn3的UIButton Outlet。 测试以下代码:

- (void)viewDidLoad
{
    [super viewDidLoad];

    NSLog(@"Btn3 %@",btn3);

    NSLog(@"viewDidLoad");
}


-(void) awakeFromNib
{
    [super awakeFromNib];

    NSLog(@"Btn3 %@",btn3);

    NSLog(@"awakeFromNib");
}

会打印:

Btn3 (null)
AwakeFromNib
Btn3 <UIRoundedRectButton: 0x64088e0; frame = (114 211; 72 37); opaque = NO; autoresize = RM+BM; layer = <CALayer: 0x6408890>>
ViewDidLoad

场景2: 删除xib文件,将UIView作为子进程添加到MainWindow.Xib中的TestingAwakeFromNibViewController,并将UIButton作为子视图添加到UIView(并将UI按钮插座连接到TestingAwakeFromNibViewController的相应插座)。

现在运行上面的代码将打印:

Btn3 <UIRoundedRectButton: 0x4e31c30; frame = (114 211; 72 37); opaque = NO; autoresize = RM+BM; layer = <CALayer: 0x4e31be0>>
viewDidLoad
Btn3 <UIRoundedRectButton: 0x4e31c30; frame = (114 211; 72 37); opaque = NO; autoresize = RM+BM; layer = <CALayer: 0x4e31be0>>
awakeFromNib

在AwakeFromNib之前表示ViewDidLoad。

第三种情景: 与第二个相同,只是没有调用[super awakeFromNib];

Btn3 <UIRoundedRectButton: 0x4e0ddf0; frame = (114 211; 72 37); opaque = NO; autoresize = RM+BM; layer = <CALayer: 0x4e0dda0>>
awakeFromNib

现在甚至没有调用ViewDidLoad。

因此,似乎不同的场景需要采取不同的行动,我们需要根据我们所采取的行动做好准备。

答案 2 :(得分:5)

我使用基于导航的应用程序选项创建一个测试项目,并将以下代码添加到rootViewController.m。

- (void)awakeFromNib {
     NSLog(@"awakeFromNib 1");
     [super awakeFromNib];
     NSLog(@"awakeFromNib 2");
}

- (void)viewDidLoad {
     NSLog(@"viewDidLoad 1");
     [super viewDidLoad];
     NSLog(@"viewDidLoad 2");
}

然后,我从控制台得到了结果:

awakeFromNib 1
awakeFromNib 2
viewDidLoad 1
viewDidLoad 2

加载控制器视图时将调用-(void)viewDidLoad。因此,当您第一次使用self.view = ...时,-(void)viewDidLoad将被调用。


如果你写了类似的内容,那么 - (void)viewDidLoad将首先被调用。

  - (void)awakeFromNib {
       NSLog(@"awakeFromNib 1");

       // The log sequence will be funny, if `viewDidLoad` raised before [super awakeFromNib]
       // If you are curios about it, just give it a try.           
       // self.view.backgroundColor = [UIColor clearColor];

       [super awakeFromNib];

       /// viewDidLoad will be called 
       /// because self.view must be loaded first.
       self.view.backgroundColor = [UIColor clearColor];  

       NSLog(@"awakeFromNib 2");
  }

并获得以下结果。

awakeFromNib 1
viewDidLoad 1
viewDidLoad 2
awakeFromNib 2

答案 3 :(得分:1)

没有专家,并且在这篇帖子之后,我意识到比在Tab控制器场景中,在“子”视图控制器中,当加载选项卡控制器(父级)时执行awakeFromNib方法,但是只有当它的“”时才执行viewDidLoad单击“标签”。

因此,据我所知,只有在选择(点击)特定标签时才能加载数据