美文网首页iOS控件使用大全iOS学习iOS Developer
自定义控件初始化代码中的坑

自定义控件初始化代码中的坑

作者: 秦砖 | 来源:发表于2016-11-26 14:26 被阅读203次

初始化代码应该是IOS开发者最先接触到的代码了。很容易、很简单是不,我也是这样觉得的。直到我使用了github上一个有1000多人star的开源控件,遇到了一个不可思议的BUG时,才发现踩进这个坑的应该不只是初学者!一般开发者最容易写出下面的初始化代码。

@interface YDTestView : UIView
-(id)initWithFrame:(CGRect)frame;
@end

@implementation YDTestView

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

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

@implementation ViewController

-(void)viewDidLoad{
  [super viewDidLoad];

  YDTestView* view = [[YDTestView alloc] init];
  [view setFrame:CGRectMake(xx, xx, xx, xx)];
  [self.view addSubview:view];
}

@end
这里看上去一点问题没有实际运行也没有什么问题,打印的LOG日志如下: init调用日志

如果你对LOG打印有疑问的话,请实际运行并思考下为什么initWithFrame接口会被调用到。

要说的坑

上面的初始化代码的坑就是出现在[self initalize]这句代码里。它负责整个控件的初始化工作,开发者会将所有的与初始化相关的动作都放置在这里。那么将kvo中的addObserver放置在initalize接口,removerObserver放置在dealloc接口中是最正常不过的选择了,如此坑就出现了。

-(void)initialize
{
  ...
  [self addObserver:self forKeyPath:BorderStyleKeyPath options:NSKeyValueObservingOptionNew context:nil];
}

-(void)dealloc
{
  ...
  [self removeObserver:self forKeyPath:BorderStyleKeyPath];
}
运行测试应用,当前页面退出时,应用崩溃了,报了下图中的异常: addObser导致的崩溃Log

但我们明明在delloc里释放了这些观察对象的。这就要和上面的两个init方法联系到一块来说了。开发者初始化时直接调用了init接口而没有使用initWithFrame接口,但init接口会默认调用一次initWithFrame接口。也就是说初始化initialize接口在一次初始化过程被调用了两次,那么addObserver接口同样被调用了两次,但仅仅释放了一次,那么出现上面的崩溃就容易理解了。

如何避免

理解了错误产生的原因,那么怎样避免就显而易见了。这里给出这样的建议,在自定义的控件的初始化代码中,如果实现了initWithFrame接口,就不要再重写init接口实现了,或者将addObserver调用放置到初始化过程之外的代码中。

后记

可能是没有清晰的表达我的想法,评论中有同学在纠结initialize这个接口。这里说明下,这个接口只是用来表达init与initWithFrame两个接口的共有部分代码,它可以叫做任何你希望的名字。这个BUG出现的原因是使用者调用init接口间接导致了addObserver被调用了两次,但在控件销毁时只removeObserver一次。

相关文章

  • 自定义控件初始化代码中的坑

    初始化代码应该是IOS开发者最先接触到的代码了。很容易、很简单是不,我也是这样觉得的。直到我使用了github上一...

  • 自定义控件交叉布局(customViewGroupCrossLa

    效果是:点击按钮使一个自定义布局里的的控件交叉 初始化一下数据:自定义控件的代码: 布局代码: 效果: 现在来处理...

  • 自定义view实现粘性拉动效果

    Chapter 2 2-1 添加自定义控件并绘制圆 新建一个自定义控件 实现基本初始化方法 在自定义控件中绘制圆代...

  • Android SearchView的自定义

    SearchView是Android原生的控件,功能较全,简单修改代码可实现自定义的效果。 首先初始化一下控件 1...

  • XIB和代码自定义控件的步骤

    title : XIB和代码自定义控件的步骤category : UI 代码和XIB自定义控件和封装子控件的步骤...

  • 如何用Swift创建自定义iOS控件

    本文将分两个部分内容:基于代码的自定义控件和基于XIB的自定义控件。 基于代码的自定义控件 实现的基本步骤(纯文字...

  • 将对象赋给cell

    //在自定义cell中添加对象属性,在.m文件中初始化cell内的自定义控件,重写setter方法//在对象的.m...

  • iOS 自定义分段控件实现及使用

    实现自定义分段控件 相关属性声明 封装初始化类方法。调用初始化方法传入参数:需要设定的整个控件的frame 以及 ...

  • iOS-视图控制器

    视图控制器指定自定义View 自定义视图类继承UIView。在初始化方法中添加子视图控件。重写controller...

  • AndroidStudio 3.0 New Features

    1.自定义控件在布局文件中使用时,preview 视窗能够根据你对自定义控件中相关代码的修改自动调整 previe...

网友评论

  • gong2012v1987:initialize方法应该是苹果预留的接口让你去实现但不需要主动调用的吧,有点类似drawrect方法吧,个人看法
  • Vine_Finer:为什么要调用initialize方法?我写了好多,都没用这个也能跑。大神求解答。

本文标题:自定义控件初始化代码中的坑

本文链接:https://www.haomeiwen.com/subject/nrtgpttx.html