UI控件初始化问题:initWithFrame和initWithCoder、aweakFromNib的执行
在iOS学习和程序开发过程中,我们经常会遇到一些自定义UI控件或控制器在初始化时出现问题,尤其在大家刚开始接触时,几种初始化方法的作用以及调用的时机往往容易混淆,这也跟我们对iOS程序设计中,类的创建和实例化的过程了解不透彻有关系。本文用一些小例子来简单梳理一下几者的关系,后面再陆续讨论一些复杂情况的深入对比。
问题:一、什么时候用initWithFrame,什么时候用aweakFromNib、initWithCoder
二、在初始化时控件自身的frame何时能获得?layoutSubViews何时调用
首先,我们实例化一个(控件类型)对象可以有多种方式:
(1)纯代码创建。创建自定义的UI控件类,然后实例化该类型的对象。
(2)通过IB(Interface Builder)创建,就是俗称的“拖线”。当我们创建好xib文件的时候,就相当于创建好了控件类,但是如果不实例化,也是没有用的,所以需要加载,这里用loadNibName来加载(实例化)UI控件。
1、搭建实验环境A,代码创建控件(TestCodeingView继承自UIView)
-(void)loadFromCoding
{
TestCodeingView * viewCoding = [[TestCodeingView alloc]init];
viewCoding.frame=CGRectMake(100,100,200,200);
viewCoding.backgroundColor=[UIColor greenColor];
[self.view addSubview:viewCoding];
}
在TestCodeingView类中对以下方法进行重写-(instancetype)init
{
self=[super init];
NSLog(@" init =====> 执行了");
NSLog(@"此时view的frame====》 %@",NSStringFromCGRect(self.frame));
return self;
}-(instancetype)initWithFrame:(CGRect)frame
{
self=[super initWithFrame:frame];
NSLog(@" initWithFrame =====> 执行了");
NSLog(@"此时view的frame====》 %@",NSStringFromCGRect(self.frame));
return self;
}-(instancetype)initWithCoder:(NSCoder *)aDecoder
{
self=[super initWithCoder:aDecoder];
NSLog(@" initWithCoder =====> 执行了");
return self;
}-(void)awakeFromNib
{
NSLog(@" awakeFromNib =====> 执行了");
}-(void)layoutSubviews
{
NSLog(@" layoutSubviews =====> 执行了");
NSLog(@"此时view的frame====》 %@",NSStringFromCGRect(self.frame));
}
运行结果:
然后更改部分代码:
-(instancetype)initWithFrame:(CGRect)frame
{
self=[super initWithFrame:frame];
NSLog(@" initWithFrame =====> 执行了");
NSLog(@"此时view的frame====》 %@",NSStringFromCGRect(self.frame));
UILabel * label = [[UILabel alloc]init];
label.text=@"我是新建的label";
label.backgroundColor=[UIColor orangeColor];
self.label=label;
[self addSubview:label];
return self;
}-(void)layoutSubviews
{
NSLog(@" layoutSubviews =====> 执行了");
NSLog(@"此时view的frame====》 %@",NSStringFromCGRect(self.frame));
self.label.frame = CGRectMake((self.frame.size.width-150)/2,self.frame.size.height/2,150,30);
}
运行结果:
小结一下:(1)纯代码创建的UI控件不执行aweakFromNib方法和 initWithCoder方法。
(2)layoutSubciews方法在控件初始化完成后(自身和子控件的实例化结束)调用,方法中能获得到当前控件的frame,以便于给子控件布局。如有子控件,调用两次。
(3)系统在调用以上方法时,有着特定的先后顺序。
2、搭建实验环境B,Xib创建控件
通过xib加载自定义UI控件,如下图,TestXibView类为手动创建的UI控件类,继承自UIView
-(void)loadFromXib
{
TestXibView * viewXib = [[[NSBundle mainBundle]loadNibNamed:@"testXibView" owner:nil options:nil] lastObject];
viewXib.center=self.view.center;
[self.view addSubview:viewXib];
}
在TestCodeingView类中对以下方法进行重写
-(instancetype)init
{
self=[super init];
NSLog(@" init =====> 执行了");
return self;
}-(instancetype)initWithFrame:(CGRect)frame
{
self=[super initWithFrame:frame];
NSLog(@" initWithFrame =====> 执行了");
return self;
}-(instancetype)initWithCoder:(NSCoder *)aDecoder
{
self=[super initWithCoder:aDecoder];
NSLog(@" initWithCoder =====> 执行了");
return self;
}-(void)awakeFromNib
{
NSLog(@" awakeFromNib =====> 执行了");
}-(void)layoutSubviews
{
NSLog(@" layoutSubviews =====> 执行了");
}
运行结果:
更改部分代码,对Xib加载的控件使用代码进行修改 (添加了一个子控件和更改背景颜色):
-(instancetype)initWithCoder:(NSCoder *)aDecoder
{
self=[super initWithCoder:aDecoder];
NSLog(@" initWithCoder =====> 执行了");
UILabel * label = [[UILabel alloc]initWithFrame:CGRectMake(0,0,150,30)];
label.text=@"我是新建的label";
label.backgroundColor=[UIColor orangeColor];
label.center=CGPointMake(self.center.x, self.frame.size.height-30);
[self addSubview:label];
return self;
}-(void)awakeFromNib
{
NSLog(@" awakeFromNib =====> 执行了");
self.backgroundColor=[UIColor yellowColor];
}
运行结果:
小结一下:(1)通过Xib创建UI控件,不会调用init和initwith方法。
(2)创建一个控件类,和xib关联,是可以修改Xib中的属性的。
(3)一样会调用layoutSubViews方法
(4)因为通过拖线和配置,已经固定了控件的大小和布局,所以frame可以获得
(5)initWithCoder和 aweakFromNib 在这里作用相同,都被系统调用
总结及延伸:
当我们弄清楚控制器加载的各种情况后,相对于用代码,使用IB和xib文件来组织UI,可以省下大量代码和时间,从而得到更快的开发速度;同时,Xib最大的问题在于其设置往往并非最终设置,在代码中你将有机会覆盖你在xib文件中进行的UI设计,造成错误和混乱。
说了好多,总结一下也无非几句话:
1、用Xib创建控件,对于控件的后续操作都写在initWithCoder或aweakFromNib方法中;
2、纯代码写创建的控件,对于控件的后续操作都写在initWithFrame方法中;
3、添加子控件时,注意布局(frame的获得),合理灵活的使用xib加载控件;
4、至于initWithCoder和aweakFromNib的区别在后面再做讨论(关于通过xib加载控制器)。
网友评论