在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(, , , );
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-)/,self.frame.size.height/, , ); }

运行结果:

小结一下:(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(, , , )];
label.text=@"我是新建的label";
label.backgroundColor=[UIColor orangeColor];
label.center=CGPointMake(self.center.x, self.frame.size.height-);
[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加载控制器)。

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

  1. 【iOS 开发】基本 UI 控件详解 (UIButton | UITextField | UITextView | UISwitch)

    博客地址 : http://blog.csdn.net/shulianghan/article/details/50051499 ; 一. UI 控件简介 1. UI 控件分类 UI 控件分类 : 活 ...

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

    转载请注明本文出自大苞米的博客(http://blog.csdn.net/a396901990),谢谢支持! Android L: Google已经确认Android L就是Android Lolli ...

  3. grootJS ui控件定义

    index13.html <html><head> <title>ui控件定义</title> <script src="jquery- ...

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

    用体感来实现UI控件的点击,如点击按钮. 做法:用一个图片表示左手手掌,图片位置追踪左手手掌移动,当手掌位于UI控件的矩形内时,握拳表示点击该控件. using UnityEngine; using ...

  5. 面试题汇总--数据储存/应用程序/UI控件/客户端的安全性与框架处理。。。

    一 数据储存  1.如果后期需要增加数据库中的字段怎么实现,如果不使用 CoreData 呢?编写 SQL 语句来操作原来表中的字段1)增加表字段ALTER TABLE 表名 ADD COLUMN 字 ...

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

    转载注明出处 : http://blog.csdn.net/shulianghan/article/details/50348982 一. 日期选择器 (UIDatePicker) UIDatePic ...

  7. 【Unity/Kinect】Kinect实现UI控件的点击

    用体感来实现UI控件的点击,如点击按钮. 做法:用一个图片表示左手手掌,图片位置追踪左手手掌移动,当手掌位于UI控件的矩形内时,握拳表示点击该控件. using UnityEngine; using ...

  8. WinForm/Silverlight多线程编程中如何更新UI控件的值

    单线程的winfom程序中,设置一个控件的值是很easy的事情,直接 this.TextBox1.value = "Hello World!";就搞定了,但是如果在一个新线程中这么 ...

  9. 富客户端 wpf, Winform 多线程更新UI控件

    前言 在富客户端的app中,如果在主线程中运行一些长时间的任务,那么应用程序的UI就不能正常相应.因为主线程要负责消息循环,相应鼠标等事件还有展现UI. 因此我们可以开启一个线程来格外处理需要长时间的 ...

随机推荐

  1. SQL Server I/O Basics

     SQL Server I/O Basics Chapter 1http://www.microsoft.com/technet/prodtechnol/sql/2000/maintain/sqlIO ...

  2. NSURLConnection和NSMutableURLRequest 实现同步、异步请求

    我是走向ios的一个小书童,我有很多不懂的,新鲜的知识去学习,去掌握! 我首先要吐槽一下: 那些不负责的博友!你分享知识本来是好事!可是你直接Control+V就是你的不对了! 尼玛,直接Contro ...

  3. Oracle EBS - 工单状态

    Job status update 1. Job的几种状态 unreleased --未核发 released--已核发 complete --完成 complete no charges--完成不计 ...

  4. linux驱动开发流程

    嵌入式linux驱动开发流程嵌入式系统中,操作系统是通过各种驱动程序来驾驭硬件设备的.设备驱动程序是操作系统内核和硬件设备之间的接口,它为应用程序屏蔽了硬件的细节,这样在应用程序看来,硬件设备只是一个 ...

  5. 转:VMware中三种网络连接的区别

    转自:http://www.cnblogs.com/rainman/archive/2013/05/06/3063925.html VMware中三种网络连接的区别   1.概述 2.bridged( ...

  6. Struts拦截器(转)

    xml <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE struts PUBLIC &qu ...

  7. Android WIFI模块分析

    一:什么是WIFI WIFI是一种无线连接技术.可用于手机.电脑.PDA等终端. WIFI技术产生的目的是改善基于IEEE802.11标准的无线网络产品之间的互通性,也就是说WIFI是基于802.11 ...

  8. 关于HTTP1.1的长连接

    HTTP是一个构建在传输层的TCP协议之上的应用层的协议,在这个层的协议,是一种网络交互须要遵守的一种协议规范. HTTP1.0的短连接 HTTP 1.0规定浏览器与server仅仅保持短暂的连接.浏 ...

  9. VUE 之 生命周期

    1. Vue实例的生命周期分为8个周期 1.1 beforeCreate:在实例创建前 <div id="app"> {{ name }} <button @cl ...

  10. 常用的sql命令

    1 mysql创建数据库 create database [database name]; 2 创建表 create table [table name]([first column name] [f ...