iOS 设置单例为nil

单例模式是各种开发语言中必然涉及到的一种设计模式,例如在iOS中有很多系统方法都是使用的单例,例如:[NSNotificationCenter defaultCenter].

 

用四人帮(GoF)教科书的说法:

Ensures a class has only one instance, and provide a global point of access to it

                                                                   单例类图

 

iOS实现单例的方式有几种,今天重点介绍使用GCD实现单例以及在使用过程中遇到的问题

GCD方式实现单例:

在单例类中定义两个属性分别是defaultValue,name;defaultValue给一个模式值,name在实例化后给赋值

step1代码实现

step1代码输出结果

如我们所想,输出instance地址以及 step1 defauleValue=init defaultValue, name=step1 name

 

接下来我们进入到第二步的测试,将instance置为nil,然后再重新实例化instance,看输入的内存地址以及属性值

step2代码实现

step2步骤的输出可能会出乎我们的想象,直接看结果

 

step2代码输出结果

和你想的是否一样呢?

按照正常的一个认知,将实例对象instance置为nil后,重新实例化后,应该是一个新的对象;但是从运行结果可以看出(见图中红色圈中内容),重新实例化后与初次实例化后的地址是一样的,也就是指向了同一块内存地址,所以属性输出内容与第一次完全一样(见图中绿色下划线内容)

那么是什么原因造成的呢?进入到dispatch_once的源码实现中去探究下

dispatch_once源码

 

注意dispatch_once源码中的红线部分,dispatch_once_t *predicate;  dispatch_once_t又是什么呢?继续看源码

dispatch_once_t源码

 

通过源码我们可以清晰的知道,dispatch_once_t是一个long类型的变量,初始化必须为0;

至此,我们可以清晰的了解到,dispatch_once的实现逻辑;

dispatch_once主要是根据long类型的值决定怎么去执行代码(所以dispatch_once_t需要声明为static,保证全局只有1个)

当值为0时,执行dispatch_once的block代码

当值为-1时,跳过dispatch_once的block代码

当值为其他值时,线程被阻塞,等待其值改变;

当dispach_once的block执行完成后,将long类型的值设置为-1.其他线程不在阻塞,跳过block。下周再调用shareInstance的时候,block已经为-1。直接跳过block。

 

线程实现观察onceToken的值


 

运行结果

 

了解了dispatch_once的实现原理后,也就了解了文章开头中将实例instance=ni后,再次实例化后,得到的还是同样的结果,那么应该如何修改呢?

关键就在与onceToken的值,添加一个方法,将onceToken置为0

 

单例完整实现

 

step3代码实现

 

step3输出结果


 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值