一 不嚴(yán)謹(jǐn)寫法
先附上不嚴(yán)謹(jǐn)?shù)膭?chuàng)建單例的寫法
- SignalModel.h
@interface SignalModel : NSObject
+ (instancetype)shareInstance;
@end
- SignalModel.m
@implementation SignalModel
+ (instancetype)shareInstance {
static SignalModel *_instance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
_instance = [[SignalModel alloc] init];
});
return _instance;
}
外界使用
SignalModel *signal1 = [[SignalModel alloc] init];
SignalModel *signal2 = [SignalModel shareInstance];
SignalModel *signal3 = [SignalModel shareInstance];
NSLog(@"\nsignal1 = %@\nsignal2 = %@\nsignal3 = %@\n",signal1,signal2,signal3);
打印結(jié)果
- 通過上面的測試微姊,可以看到通過
shareInstance
方法獲取的對象是相同的酸茴,但是用alloc
和init
構(gòu)造對象時,得到的對象卻是不一樣的柒桑。- 通過不同的方式獲得不同的對象弊决,是有問題的,所以要封鎖初始化的方式魁淳,如
alloc
飘诗,copy
,mutableCopy
界逛,new
摘抄的原理哈
創(chuàng)建對象的步驟分為申請內(nèi)存(alloc)
昆稿、初始化(init)
這兩個步驟,我們要確保對象的唯一性息拜,因此在第一步這個階段我們就要攔截它溉潭。當(dāng)我們調(diào)用alloc方法時净响,OC內(nèi)部會調(diào)用allocWithZone
這個方法來申請內(nèi)存,我們覆寫這個方法喳瓣,然后在這個方法中調(diào)用shareInstance方法返回單例對象馋贤,這樣就可以達(dá)到我們的目的∥飞拢拷貝對象也是同樣的原理配乓,覆寫copyWithZone
方法,然后在這個方法中調(diào)用shareInstance方法返回單例對象
二 正確寫法
- SignalModel.m
// 實(shí)現(xiàn)copy協(xié)議
@interface SignalModel()<NSCopying, NSMutableCopying>
@end
+ (instancetype)shareInstance {
static SignalModel *_instance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
_instance = [[super allocWithZone:NULL] init];
});
return _instance;
}
+ (instancetype)allocWithZone:(struct _NSZone *)zone {
return [self shareInstance];
}
- (id)copyWithZone:(NSZone *)zone {
return self;
}
- (id)mutableCopyWithZone:(NSZone *)zone {
return self;
}
shareInstance
單例方法中惠毁,變量的初始化改成[[super allocWithZone:NULL] init]
- 實(shí)現(xiàn)
copyWithZone:
和mutableCopyWithZone:
方法
測試代碼
SignalModel *signal1 = [[SignalModel alloc] init];
SignalModel *signal2 = [SignalModel shareInstance];
SignalModel *signal3 = [SignalModel shareInstance];
SignalModel *signal4 = [SignalModel new];
SignalModel *signal5 = [signal1 copy];
SignalModel *signal6 = [signal2 mutableCopy];
NSLog(@"\nsignal1 = %@\nsignal2 = %@\nsignal3 = %@\nsignal4 = %@\nsignal5 = %@\nsignal6 = %@",signal1,signal2,signal3,signal4,signal5,signal6);
打印結(jié)果
無論通過哪種方式創(chuàng)建出來的實(shí)例對象犹芹,其內(nèi)存地址都是一樣的,所以該種寫法才是嚴(yán)謹(jǐn)?shù)摹?/p>