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加载控制器)。

时间: 2024-12-14 18:15:46

UI控件初始化问题:initWithFrame和initWithCoder、aweakFromNib的执行的相关文章

iOS UI控件7(UITableView)

1.表格(UITableView)与表格控制器(UITableViewController) UITableView是iOS开发中常见的UI控件,本质是一个列表(单列的表格).UITableView允许自由控制行的控件,包括在表格行中添加多个字控件.UITableView继承了UIScrollView,具有UIScrollView的功能,这个UIScrollView主要封装了UITableViewCell单元格控件,因此UITableView默认可以对单元格进行滚动.默认情况下,所有UITabl

iOS UI控件6

1.微调器(UIStepper) iOS5 新增UI,包含 +.-两个按钮,继承了UIControl 支持的属性: Value Minimum Maximum Current Step Behavior Autorepeat 按住 加号 不松手,数字会持续变化 Continuous 为YES时,用户交互会立即出发ValueChanged事件,NO 表示只有等用户交互结束才出发ValueChanged事件 Wrap 若为YES,value加到超过Maximum值时,会变成Min指. 设置自定义外观

ANDROID L——Material Design详解(UI控件)

转载请注明本文出自大苞米的博客(http://blog.csdn.net/a396901990),谢谢支持! Android L: Google已经确认Android L就是Android Lollipop(5.0). 前几天发现Android5.0正式版的sdk已经可以下载了,而且首次搭载Android L系统的Nexus 6和 Nexus 9也即将上市. 所以是时候开始学习Android L了! 关于Android L如何配置模拟器和创建项目,如果大家有兴趣的话可以看看我之前的一篇文章: A

IOS Ui控件 修改位置和尺寸,代码添加控件

所有的UI控件最终都继承自UIView,UI控件的公共属性都定义在UIView中, UIView的常见属性 UIView *superview; 获得自己的父控件对象 NSArray *subviews; 获得自己的所有子控件对象 NSInteger tag; 控件的ID(标识),父控件可以通过tag来找到对应的子控件 CGAffineTransform transform; 控件的形变属性(可以设置旋转角度.比例缩放.平移等属性) CGRect frame; 控件所在矩形框在父控件中的位置和尺

初级篇第四期:纯代码来写UI控件

学习建议:自己动手,丰衣足食 学习周期:1周 学习目的:熟练使用Obejct-C中最常用的简单UI控件 学习答疑:欢迎来技术群里提问并做分享 学习工具:Xcode开发环境 学习内容:我们会在下面告诉大家手动创建常见的UI控件 经过前几期的学习,相信小伙伴们应该对UI控件有所熟悉了哦,没错,那我们接下来就用纯代码来写一些常用的UI控件好了 首先,任何一个UI控件都是有它固定的属性的,第一就是frame,一定要记住,没有frame就没有它的存在,所以一个UI控件的灵魂就在于它的frame,因为它是显

【IOS 开发】基本 UI 控件详解 (UIDatePicker | UIPickerView | UIStepper | UIWebView | UIToolBar )

转载注明出处 : http://blog.csdn.net/shulianghan/article/details/50348982 一. 日期选择器 (UIDatePicker) UIDatePicker 属性截图 : 1. UIDatePicker 控件属性 (1) Mode 属性 Mode 属性 : 用于设置 UIDatePicker 模式; -- Date 属性值 : 显示日期, 不显示时间; -- Time 属性值 : 显示时间, 不显示日期; -- Date and Time 属性值

Kinect用体感来实现UI控件的点击

用体感来实现UI控件的点击,如点击按钮. 做法:用一个图片表示左手手掌,图片位置追踪左手手掌移动,当手掌位于UI控件的矩形内时,握拳表示点击该控件. using UnityEngine; using System.Collections; using UnityEngine.UI; /// <summary> /// 用Kinect实现UI的点击:追踪左手手掌,移动到UI矩形内握拳表示点击该按钮. /// </summary> public class UIClick : Mono

UI控件懒加载问题01

UI 控件懒加载问题: 1, 什么时候使用懒加载加载UI控件? 2, 加载控件的什么属性? 3, 用什么类型的指针修饰控件? code : (ARC) 定义属性, @property(nonatomic,weak) UIButton *customBtn1;   重写getter方法 -(UIButton *)customBtn1{ if (_customBtn1 == nil) { _customBtn1 = [UIButton buttonWithType:UIButtonTypeCusto

UI控件 UIActivityIndicatorView 等待时出现的动态旋转图,以及自定义颜色

#import "ViewController.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad { //    UIActivityIndicatorView  等待旋转 [super viewDidLoad]; UIButton * btn = [[UIButton alloc]initWithFrame:CGRectMake(100, 100, 140,