目录
- 一、常见的单例及可能存在的问题
- 二、完善的单例
- 三、可继承的单例
- 四、单例的销毁
前言
你是不是觉得这是一个老生常谈的问题,一个你做开发想说自己没接触过都不可能的问题,有什么好说的?对的,你说的没错,但是你真的对他完全掌握了吗?简单的问你几个问题:
- 1、怎么保证创建的单例类使用它的任何创建方法都不会有新的实例生成?
- 2、单例怎么销毁?
- 3、单例怎么继承?
如果你全都清楚了,那请略过本篇文章。如果还有不清楚的,请往下看。
一、常见的单例及可能存在的问题
常见的单例类如下:
+ (NormalSharedInstanceClass *)sharedInstance {
static NormalSharedInstanceClass *_sharedInstance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
_sharedInstance = [[self alloc] init];
});
return _sharedInstance;
}
测试该类是否存在问题的代码如下:
- (void)printNormalSharedInstance {
NormalSharedInstanceClass *single1 = [NormalSharedInstanceClass sharedInstance];
NSString *singleMessage1 = [NSString stringWithFormat:@"[1]sharedInstance = %@", single1];
NSLog(@"%@", singleMessage1);
NormalSharedInstanceClass *single2 = [NormalSharedInstanceClass sharedInstance];
NSString *singleMessage2 = [NSString stringWithFormat:@"[2]sharedInstance = %@", single2];
NSLog(@"%@", singleMessage2);
NormalSharedInstanceClass *single3 = [[NormalSharedInstanceClass alloc] init];
NSString *singleMessage3 = [NSString stringWithFormat:@"[3][[xxx alloc] init] = %@", single3];
NSLog(@"%@", singleMessage3);
NormalSharedInstanceClass *single4 = [single3 copy];
NSString *singleMessage4 = [NSString stringWithFormat:@"[4]copy = %@", single4];
NSLog(@"%@", singleMessage4);
}
打印测试结果如下:
[1]sharedInstance = <NormalSharedInstanceClass: 0x6000007911e0>
[2]sharedInstance = <NormalSharedInstanceClass: 0x6000007911e0>
[3][[xxx alloc] init] = <NormalSharedInstanceClass: 0x60000079c0c0>
[NormalSharedInstanceClass copyWithZone:]: unrecognized selector sent to instance 0x60000079c0c0
测试结果分析:
当我们调用shareInstance方法时获取到的对象是相同的;
但是当我们通过alloc和init来构造对象的时候,得到的对象却是不一样的;
而且在 single4 这行注释打开后, 会在此崩溃。
所以,当我们通过不同的途径获取对象,并不能保证对象的唯一性,,所以我们就需要封锁用户通过alloc和init以及copy来构造对象这条道路。
二、完善的单例
完善的单例代码如下:
+ (PerfectSharedInstanceClass *)sharedInstance {
static PerfectSharedInstanceClass *_sharedInstance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
//_sharedInstance = [[self alloc] init];
_sharedInstance = [[super allocWithZone:NULL] init];
});
return _sharedInstance;
}
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
return [self sharedInstance];
}
- (id)copyWithZone:(nullable NSZone *)zone {
return [PerfectSharedInstanceClass sharedInstance];
}
测试该类是否存在问题的代码如下:
- (void)printPerfectSharedInstance {
PerfectSharedInstanceClass *single1 = [PerfectSharedInstanceClass sharedInstance];
NSString *singleMessage1 = [NSString stringWithFormat:@"[1]sharedInstance = %@", single1];
NSLog(@"%@", singleMessage1);
PerfectSharedInstanceClass *single2 = [PerfectSharedInstanceClass sharedInstance];
NSString *singleMessage2 = [NSString stringWithFormat:@"[2]sharedInstance = %@", single2];
NSLog(@"%@", singleMessage2);
PerfectSharedInstanceClass *single3 = [[PerfectSharedInstanceClass alloc] init];
NSString *singleMessage3 = [NSString stringWithFormat:@"[3][[xxx alloc] init] = %@", single3];
NSLog(@"%@", singleMessage3);
PerfectSharedInstanceClass *single4 = [single3 copy];
NSString *singleMessage4 = [NSString stringWithFormat:@"[4]copy = %@", single4];
NSLog(@"%@", singleMessage4);
}
打印测试结果如下:
[1]sharedInstance = <PerfectSharedInstanceClass: 0x60000057e030>
[2]sharedInstance = <PerfectSharedInstanceClass: 0x60000057e030>
[3][[xxx alloc] init] = <PerfectSharedInstanceClass: 0x60000057e030>
[4]copy = <PerfectSharedInstanceClass: 0x60000057e030>
三、可继承的单例
1、平时的单例继承存在的问题
先来看看平时的单例的输出:
- (void)printSubNormalSharedInstance {
NormalSharedInstanceClass *single1 = [NormalSharedInstanceClass sharedInstance];
NSString *singleMessage1 = [NSString stringWithFormat:@"[1]super sharedInstance = %@", single1];
NSLog(@"%@", singleMessage1);
SubNormalSharedInstanceClass *single2 = [SubNormalSharedInstanceClass sharedInstance];
NSString *singleMessage2 = [NSString stringWithFormat:@"[2]sub sharedInstance = %@", single2];
NSLog(@"%@", singleMessage2);
}
打印的测试结果如下:
[1]super sharedInstance = <NormalSharedInstanceClass: 0x6000002c4ff0>
[2]sub sharedInstance = <NormalSharedInstanceClass: 0x6000002c4ff0>
2、可继承的单例的继承实现
可继承的单例类:
#import "InheritableSharedInstanceClass.h"
#import <objc/runtime.h>
@implementation InheritableSharedInstanceClass
+ (instancetype)sharedInstance {
id sharedInstance = objc_getAssociatedObject(self, @"cjSharedInstance");
if (!sharedInstance) {
sharedInstance = [[super allocWithZone:NULL] init];
objc_setAssociatedObject(self, @"cjSharedInstance", sharedInstance, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
return sharedInstance;
}
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
return [self sharedInstance];
}
- (id)copyWithZone:(nullable NSZone *)zone {
return [[self class] sharedInstance];
}
@end
打印的测试结果如下:
[1]super sharedInstance = <InheritableSharedInstanceClass: 0x6000002c6410>
[2]sub sharedInstance = <SubInheritableSharedInstanceClass: 0x6000002c6420>
四、单例的销毁
前面讲了单例的创建,但是有个别情况需要销毁单例.
下面分别从两种创建方法对应两种销毁形式.
//必须把static dispatch_once_t onceToken; 这个拿到函数体外,成为全局的.
+ (void)attempDealloc {
onceToken = 0; // 只有置成0,GCD才会认为它从未执行过.它默认为0,这样才能保证下次再次调用shareInstance的时候,再次创建对象.
_sharedInstance = nil;
}
End
结束!如有问题,欢迎评论!