Objective-c 中如何重写父类的初始化方法

在我们的日常开发中我们经常会定义一些自己的子类继承一些UIKit 库中的类,那我们应该如何重写的这些初化方法呢?那我们先看看这些类有哪些初初化方法吧。(这里就用UIView为例)

- (id)init;
- (nullable instancetype)initWithCoder:(NSCoder *)aDecoder NS_DESIGNATED_INITIALIZER;
- (instancetyp)initWithFrame:(CGRect)frame NS_DESIGNATED_INITIALIZER

  我们先说说这个几个方法的执行顺序吧,init 方法我们知道它是基类NSObject 类中继承过来,应该是最基本的方法了,返回一个自己的对象。initWithCoder 这个是我们用IB初始View来调用的。-initWithFrame方法呢我们暂时先不说先往下看。
  有时候我们经常写一些的自定义东西我们想把这些东西开源出去,(当然我们很多时候都在用开源东西)。我们就想写的很完美,我们就会重载所有的初始方法,我们先来定义一个UIView 子类MyView然后重写这些方法

#import "MyView.h"

@implementation MyView{
   UIView *subView;
}

- (id)initWithFrame:(CGRect)frame {
    self = [super initWithFrame:frame];
    if (self) {
    [self setup];
    NSLog(@"初始化-------%@",NSStringFromSelector(_cmd));
    }
   return self;
}

- (id)init {
      self = [super init];
      if (self) {
       [self setup];
        NSLog(@"初始化-------%@",NSStringFromSelector(_cmd));
  }
     return self;
}

 - (id)initWithCoder:(NSCoder *)aDecoder {
     self = [super initWithCoder:aDecoder];
     if (self) {
     NSLog(@"初始化-------%@",NSStringFromSelector(_cmd));
     }
     return self;
}
 - (void)setup {
        subView = [[UIView alloc] initWithFrame:CGRectMake(10, 10, 20, 20)];
        subView.backgroundColor = [UIColor redColor];
        [self addSubview:subView];
  }
 @end

我们先看看在IB中创建一个view 把class 设成MyView 看看方法的调用情况:


1.1.png


我们跑下程序看看输出情况是不是像我们所说的那样:


1.2


输出结果是调用了 initWithCoder.

所以如果我们想让我们的类支持IB我们可以重写这个方法然后初始化一些东西。

下面我们来试试的手动添加的我们先用initWithFrame方法:

 #import "ViewController.h"
 #import "MyView.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
  [super viewDidLoad];
  // Do any additional setup after loading the view, typically from a nib.

  MyView *myView = [[MyView alloc] initWithFrame:CGRectMake(10, 20, 100, 100)];
  myView.backgroundColor = [UIColor purpleColor];
  [self.view addSubview:myView];

}

 - (void)didReceiveMemoryWarning {
   [super didReceiveMemoryWarning];
   // Dispose of any resources that can be recreated.
}
@end

我们再来运行一下看看结果:


1.3


也符合我们的预期。那我们来看看最后一种情况了,因为我们现在的项目中越来越多人采用AutoLayout了初始化的时候可能调用下init就完了。不会去调用initWithFrame方法,那我们看看我们在重写了这三个方法后调用init初始化会出现什么情况?我稍加修改一下上面的代码:

- (void)viewDidLoad {
  [super viewDidLoad];
  // Do any additional setup after loading the view, typically from a nib.

  MyView *myView = [[MyView alloc] init];
  myView.frame = CGRectMake(10, 20, 100, 100);
  myView.backgroundColor = [UIColor purpleColor];
  [self.view addSubview:myView];
}

看看输出结果是什么?


1.4


奇怪了竟然先输出了initWithFrame ,然后才输出了init 为什么会这样子呢?如果你看过我的上篇文章你就应该懂了,我们可以回到最上面看看发现这个两个方法后面的都有NS_DESIGNATED_INITIALIZER这么一行字。什么意思呢?Objective-C 有指定初始化方法(designated initializer)和间接(secondary initializer)初始化方法的观念。 designated 初始化方法是提供所有的参数,secondary 初始化方法是一个或多个,并且提供一个或者更多的默认参数来调用 designated 初始化的初始化方法。由此我们可以看出init 应该是个secondary initializer 初始方法,当我们调用 [super init] 时候父类应该是去调用designated initializer 方法 initWithFrame 方法。所以我们不应该在我们的类里去重写secondary initializer 方法。如果像这样子都重写了那就会调用两遍我们的setup方法。很显然这样子是没有必要的。同时也会出现问题如果像我们上面那样的写法我们就会添加两个subView到同一个地方,这显然不是我们想要的结果所以我们应该避免这样的情况出现。这里我打下结果出来的可以自已尝试下看看:


1.5


所以当我们定义一个子类时:

  1. 不需要重载任何初始化函数(当然这个情况不太常用,我们要初始化一些我们自己东西)
  2. 重载 designated initializer(上面的我们只要重写initWithFrame 方法即可,如果要支持IB再重写initWithCoder 就可以了,完全没有必要再去重写init 当然你可以只重写的 init 不重写initWithFrame这样子也不会出现二次调用的问题,但这样子使用者可以使用initWithFrame方法初始化这样子就会导致一些东西会被没有初化)
  3. 定义一个新的 designated initializer
    以此类推读者可以尝试的看看UIViewController 的。(这篇文章算是上一篇文章的补充吧)。
原文地址:https://www.cnblogs.com/Free-Thinker/p/7089442.html